katex 0.10.1 → 0.12.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +141 -0
- package/LICENSE +1 -1
- package/README.md +6 -6
- package/cli.js +0 -0
- package/contrib/auto-render/auto-render.js +12 -3
- package/contrib/copy-tex/README.md +3 -5
- package/contrib/mathtex-script-type/README.md +12 -14
- package/contrib/mhchem/README.md +3 -1
- package/contrib/render-a11y-string/render-a11y-string.js +712 -0
- package/contrib/render-a11y-string/test/render-a11y-string-spec.js +526 -0
- package/dist/README.md +6 -6
- package/dist/contrib/auto-render.js +14 -3
- package/dist/contrib/auto-render.min.js +1 -1
- package/dist/contrib/auto-render.mjs +14 -3
- package/dist/contrib/mhchem.min.js +1 -1
- package/dist/contrib/render-a11y-string.js +870 -0
- package/dist/contrib/render-a11y-string.min.js +1 -0
- package/dist/contrib/render-a11y-string.mjs +753 -0
- package/dist/fonts/KaTeX_AMS-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_AMS-Regular.woff +0 -0
- package/dist/fonts/KaTeX_AMS-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Bold.ttf +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Bold.woff +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Bold.woff2 +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Caligraphic-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Fraktur-Bold.ttf +0 -0
- package/dist/fonts/KaTeX_Fraktur-Bold.woff +0 -0
- package/dist/fonts/KaTeX_Fraktur-Bold.woff2 +0 -0
- package/dist/fonts/KaTeX_Fraktur-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Fraktur-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Fraktur-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Main-Bold.ttf +0 -0
- package/dist/fonts/KaTeX_Main-Bold.woff +0 -0
- package/dist/fonts/KaTeX_Main-Bold.woff2 +0 -0
- package/dist/fonts/KaTeX_Main-BoldItalic.ttf +0 -0
- package/dist/fonts/KaTeX_Main-BoldItalic.woff +0 -0
- package/dist/fonts/KaTeX_Main-BoldItalic.woff2 +0 -0
- package/dist/fonts/KaTeX_Main-Italic.ttf +0 -0
- package/dist/fonts/KaTeX_Main-Italic.woff +0 -0
- package/dist/fonts/KaTeX_Main-Italic.woff2 +0 -0
- package/dist/fonts/KaTeX_Main-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Main-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Main-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Math-BoldItalic.ttf +0 -0
- package/dist/fonts/KaTeX_Math-BoldItalic.woff +0 -0
- package/dist/fonts/KaTeX_Math-BoldItalic.woff2 +0 -0
- package/dist/fonts/KaTeX_Math-Italic.ttf +0 -0
- package/dist/fonts/KaTeX_Math-Italic.woff +0 -0
- package/dist/fonts/KaTeX_Math-Italic.woff2 +0 -0
- package/dist/fonts/KaTeX_SansSerif-Bold.ttf +0 -0
- package/dist/fonts/KaTeX_SansSerif-Bold.woff +0 -0
- package/dist/fonts/KaTeX_SansSerif-Bold.woff2 +0 -0
- package/dist/fonts/KaTeX_SansSerif-Italic.ttf +0 -0
- package/dist/fonts/KaTeX_SansSerif-Italic.woff +0 -0
- package/dist/fonts/KaTeX_SansSerif-Italic.woff2 +0 -0
- package/dist/fonts/KaTeX_SansSerif-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_SansSerif-Regular.woff +0 -0
- package/dist/fonts/KaTeX_SansSerif-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Script-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Script-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Script-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Size1-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Size1-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Size1-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Size2-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Size2-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Size2-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Size3-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Size3-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Size3-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Size4-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Size4-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Size4-Regular.woff2 +0 -0
- package/dist/fonts/KaTeX_Typewriter-Regular.ttf +0 -0
- package/dist/fonts/KaTeX_Typewriter-Regular.woff +0 -0
- package/dist/fonts/KaTeX_Typewriter-Regular.woff2 +0 -0
- package/dist/katex.css +34 -10
- package/dist/katex.js +2906 -2115
- package/dist/katex.min.css +1 -1
- package/dist/katex.min.js +1 -1
- package/dist/katex.mjs +2809 -2020
- package/package.json +12 -11
- package/src/Lexer.js +1 -0
- package/src/MacroExpander.js +39 -10
- package/src/Options.js +15 -75
- package/src/Parser.js +152 -115
- package/src/Settings.js +70 -7
- package/src/Token.js +2 -0
- package/src/buildCommon.js +24 -90
- package/src/buildHTML.js +31 -31
- package/src/buildMathML.js +52 -9
- package/src/buildTree.js +13 -6
- package/src/defineFunction.js +7 -22
- package/src/delimiter.js +66 -27
- package/src/domTree.js +71 -4
- package/src/environments/array.js +235 -25
- package/src/fontMetrics.js +11 -2
- package/src/functions/accent.js +9 -9
- package/src/functions/accentunder.js +2 -2
- package/src/functions/arrow.js +15 -5
- package/src/functions/color.js +9 -38
- package/src/functions/def.js +184 -0
- package/src/functions/delimsizing.js +32 -8
- package/src/functions/enclose.js +33 -6
- package/src/functions/font.js +4 -1
- package/src/functions/genfrac.js +39 -27
- package/src/functions/horizBrace.js +6 -7
- package/src/functions/href.js +16 -0
- package/src/functions/html.js +102 -0
- package/src/functions/includegraphics.js +153 -0
- package/src/functions/lap.js +4 -7
- package/src/functions/math.js +1 -5
- package/src/functions/mclass.js +41 -2
- package/src/functions/op.js +27 -111
- package/src/functions/operatorname.js +136 -92
- package/src/functions/ordgroup.js +1 -1
- package/src/functions/overline.js +3 -2
- package/src/functions/phantom.js +5 -2
- package/src/functions/raisebox.js +4 -16
- package/src/functions/rule.js +20 -9
- package/src/functions/styling.js +0 -9
- package/src/functions/supsub.js +27 -7
- package/src/functions/symbolsOp.js +4 -0
- package/src/functions/tag.js +20 -4
- package/src/functions/text.js +4 -3
- package/src/functions/underline.js +3 -2
- package/src/functions/utils/assembleSupSub.js +110 -0
- package/src/functions.js +3 -0
- package/src/katex.less +45 -9
- package/src/macros.js +259 -98
- package/src/mathMLTree.js +6 -4
- package/src/parseNode.js +37 -57
- package/src/stretchy.js +3 -1
- package/src/svgGeometry.js +136 -44
- package/src/symbols.js +52 -69
- package/src/tree.js +2 -2
- package/src/types.js +2 -1
- package/src/unicodeAccents.js +3 -1
- package/src/unicodeSymbols.js +30 -321
- package/src/utils.js +10 -0
- package/src/wide-character.js +2 -2
- package/src/unicodeMake.js +0 -70
|
@@ -0,0 +1,712 @@
|
|
|
1
|
+
// @flow
|
|
2
|
+
/**
|
|
3
|
+
* renderA11yString returns a readable string.
|
|
4
|
+
*
|
|
5
|
+
* In some cases the string will have the proper semantic math
|
|
6
|
+
* meaning,:
|
|
7
|
+
* renderA11yString("\\frac{1}{2}"")
|
|
8
|
+
* -> "start fraction, 1, divided by, 2, end fraction"
|
|
9
|
+
*
|
|
10
|
+
* However, other cases do not:
|
|
11
|
+
* renderA11yString("f(x) = x^2")
|
|
12
|
+
* -> "f, left parenthesis, x, right parenthesis, equals, x, squared"
|
|
13
|
+
*
|
|
14
|
+
* The commas in the string aim to increase ease of understanding
|
|
15
|
+
* when read by a screenreader.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
// NOTE: since we're importing types here these files won't actually be
|
|
19
|
+
// included in the build.
|
|
20
|
+
import type {Atom} from "../../src/symbols";
|
|
21
|
+
import type {AnyParseNode} from "../../src/parseNode";
|
|
22
|
+
import type {SettingsOptions} from "../../src/Settings";
|
|
23
|
+
|
|
24
|
+
// $FlowIgnore: we import the types directly anyways
|
|
25
|
+
import katex from "katex";
|
|
26
|
+
|
|
27
|
+
const stringMap = {
|
|
28
|
+
"(": "left parenthesis",
|
|
29
|
+
")": "right parenthesis",
|
|
30
|
+
"[": "open bracket",
|
|
31
|
+
"]": "close bracket",
|
|
32
|
+
"\\{": "left brace",
|
|
33
|
+
"\\}": "right brace",
|
|
34
|
+
"\\lvert": "open vertical bar",
|
|
35
|
+
"\\rvert": "close vertical bar",
|
|
36
|
+
"|": "vertical bar",
|
|
37
|
+
"\\uparrow": "up arrow",
|
|
38
|
+
"\\Uparrow": "up arrow",
|
|
39
|
+
"\\downarrow": "down arrow",
|
|
40
|
+
"\\Downarrow": "down arrow",
|
|
41
|
+
"\\updownarrow": "up down arrow",
|
|
42
|
+
"\\leftarrow": "left arrow",
|
|
43
|
+
"\\Leftarrow": "left arrow",
|
|
44
|
+
"\\rightarrow": "right arrow",
|
|
45
|
+
"\\Rightarrow": "right arrow",
|
|
46
|
+
"\\langle": "open angle",
|
|
47
|
+
"\\rangle": "close angle",
|
|
48
|
+
"\\lfloor": "open floor",
|
|
49
|
+
"\\rfloor": "close floor",
|
|
50
|
+
"\\int": "integral",
|
|
51
|
+
"\\intop": "integral",
|
|
52
|
+
"\\lim": "limit",
|
|
53
|
+
"\\ln": "natural log",
|
|
54
|
+
"\\log": "log",
|
|
55
|
+
"\\sin": "sine",
|
|
56
|
+
"\\cos": "cosine",
|
|
57
|
+
"\\tan": "tangent",
|
|
58
|
+
"\\cot": "cotangent",
|
|
59
|
+
"\\sum": "sum",
|
|
60
|
+
"/": "slash",
|
|
61
|
+
",": "comma",
|
|
62
|
+
".": "point",
|
|
63
|
+
"-": "negative",
|
|
64
|
+
"+": "plus",
|
|
65
|
+
"~": "tilde",
|
|
66
|
+
":": "colon",
|
|
67
|
+
"?": "question mark",
|
|
68
|
+
"'": "apostrophe",
|
|
69
|
+
"\\%": "percent",
|
|
70
|
+
" ": "space",
|
|
71
|
+
"\\ ": "space",
|
|
72
|
+
"\\$": "dollar sign",
|
|
73
|
+
"\\angle": "angle",
|
|
74
|
+
"\\degree": "degree",
|
|
75
|
+
"\\circ": "circle",
|
|
76
|
+
"\\vec": "vector",
|
|
77
|
+
"\\triangle": "triangle",
|
|
78
|
+
"\\pi": "pi",
|
|
79
|
+
"\\prime": "prime",
|
|
80
|
+
"\\infty": "infinity",
|
|
81
|
+
"\\alpha": "alpha",
|
|
82
|
+
"\\beta": "beta",
|
|
83
|
+
"\\gamma": "gamma",
|
|
84
|
+
"\\omega": "omega",
|
|
85
|
+
"\\theta": "theta",
|
|
86
|
+
"\\sigma": "sigma",
|
|
87
|
+
"\\lambda": "lambda",
|
|
88
|
+
"\\tau": "tau",
|
|
89
|
+
"\\Delta": "delta",
|
|
90
|
+
"\\delta": "delta",
|
|
91
|
+
"\\mu": "mu",
|
|
92
|
+
"\\rho": "rho",
|
|
93
|
+
"\\nabla": "del",
|
|
94
|
+
"\\ell": "ell",
|
|
95
|
+
"\\ldots": "dots",
|
|
96
|
+
// TODO: add entries for all accents
|
|
97
|
+
"\\hat": "hat",
|
|
98
|
+
"\\acute": "acute",
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
const powerMap = {
|
|
102
|
+
"prime": "prime",
|
|
103
|
+
"degree": "degrees",
|
|
104
|
+
"circle": "degrees",
|
|
105
|
+
"2": "squared",
|
|
106
|
+
"3": "cubed",
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
const openMap = {
|
|
110
|
+
"|": "open vertical bar",
|
|
111
|
+
".": "",
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
const closeMap = {
|
|
115
|
+
"|": "close vertical bar",
|
|
116
|
+
".": "",
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
const binMap = {
|
|
120
|
+
"+": "plus",
|
|
121
|
+
"-": "minus",
|
|
122
|
+
"\\pm": "plus minus",
|
|
123
|
+
"\\cdot": "dot",
|
|
124
|
+
"*": "times",
|
|
125
|
+
"/": "divided by",
|
|
126
|
+
"\\times": "times",
|
|
127
|
+
"\\div": "divided by",
|
|
128
|
+
"\\circ": "circle",
|
|
129
|
+
"\\bullet": "bullet",
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
const relMap = {
|
|
133
|
+
"=": "equals",
|
|
134
|
+
"\\approx": "approximately equals",
|
|
135
|
+
"≠": "does not equal",
|
|
136
|
+
"\\geq": "is greater than or equal to",
|
|
137
|
+
"\\ge": "is greater than or equal to",
|
|
138
|
+
"\\leq": "is less than or equal to",
|
|
139
|
+
"\\le": "is less than or equal to",
|
|
140
|
+
">": "is greater than",
|
|
141
|
+
"<": "is less than",
|
|
142
|
+
"\\leftarrow": "left arrow",
|
|
143
|
+
"\\Leftarrow": "left arrow",
|
|
144
|
+
"\\rightarrow": "right arrow",
|
|
145
|
+
"\\Rightarrow": "right arrow",
|
|
146
|
+
":": "colon",
|
|
147
|
+
};
|
|
148
|
+
|
|
149
|
+
const accentUnderMap = {
|
|
150
|
+
"\\underleftarrow": "left arrow",
|
|
151
|
+
"\\underrightarrow": "right arrow",
|
|
152
|
+
"\\underleftrightarrow": "left-right arrow",
|
|
153
|
+
"\\undergroup": "group",
|
|
154
|
+
"\\underlinesegment": "line segment",
|
|
155
|
+
"\\utilde": "tilde",
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
type NestedArray<T> = Array<T | NestedArray<T>>;
|
|
159
|
+
|
|
160
|
+
const buildString = (
|
|
161
|
+
str: string,
|
|
162
|
+
type: Atom | "normal",
|
|
163
|
+
a11yStrings: NestedArray<string>,
|
|
164
|
+
) => {
|
|
165
|
+
if (!str) {
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
let ret;
|
|
170
|
+
|
|
171
|
+
if (type === "open") {
|
|
172
|
+
ret = str in openMap ? openMap[str] : stringMap[str] || str;
|
|
173
|
+
} else if (type === "close") {
|
|
174
|
+
ret = str in closeMap ? closeMap[str] : stringMap[str] || str;
|
|
175
|
+
} else if (type === "bin") {
|
|
176
|
+
ret = binMap[str] || str;
|
|
177
|
+
} else if (type === "rel") {
|
|
178
|
+
ret = relMap[str] || str;
|
|
179
|
+
} else {
|
|
180
|
+
ret = stringMap[str] || str;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// If the text to add is a number and there is already a string
|
|
184
|
+
// in the list and the last string is a number then we should
|
|
185
|
+
// combine them into a single number
|
|
186
|
+
if (
|
|
187
|
+
/^\d+$/.test(ret) &&
|
|
188
|
+
a11yStrings.length > 0 &&
|
|
189
|
+
// TODO(kevinb): check that the last item in a11yStrings is a string
|
|
190
|
+
// I think we might be able to drop the nested arrays, which would make
|
|
191
|
+
// this easier to type - $FlowFixMe
|
|
192
|
+
/^\d+$/.test(a11yStrings[a11yStrings.length - 1])
|
|
193
|
+
) {
|
|
194
|
+
a11yStrings[a11yStrings.length - 1] += ret;
|
|
195
|
+
} else if (ret) {
|
|
196
|
+
a11yStrings.push(ret);
|
|
197
|
+
}
|
|
198
|
+
};
|
|
199
|
+
|
|
200
|
+
const buildRegion = (
|
|
201
|
+
a11yStrings: NestedArray<string>,
|
|
202
|
+
callback: (regionStrings: NestedArray<string>) => void,
|
|
203
|
+
) => {
|
|
204
|
+
const regionStrings: NestedArray<string> = [];
|
|
205
|
+
a11yStrings.push(regionStrings);
|
|
206
|
+
callback(regionStrings);
|
|
207
|
+
};
|
|
208
|
+
|
|
209
|
+
const handleObject = (
|
|
210
|
+
tree: AnyParseNode,
|
|
211
|
+
a11yStrings: NestedArray<string>,
|
|
212
|
+
atomType: Atom | "normal",
|
|
213
|
+
) => {
|
|
214
|
+
// Everything else is assumed to be an object...
|
|
215
|
+
switch (tree.type) {
|
|
216
|
+
case "accent": {
|
|
217
|
+
buildRegion(a11yStrings, (a11yStrings) => {
|
|
218
|
+
buildA11yStrings(tree.base, a11yStrings, atomType);
|
|
219
|
+
a11yStrings.push("with");
|
|
220
|
+
buildString(tree.label, "normal", a11yStrings);
|
|
221
|
+
a11yStrings.push("on top");
|
|
222
|
+
});
|
|
223
|
+
break;
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
case "accentUnder": {
|
|
227
|
+
buildRegion(a11yStrings, (a11yStrings) => {
|
|
228
|
+
buildA11yStrings(tree.base, a11yStrings, atomType);
|
|
229
|
+
a11yStrings.push("with");
|
|
230
|
+
buildString(accentUnderMap[tree.label], "normal", a11yStrings);
|
|
231
|
+
a11yStrings.push("underneath");
|
|
232
|
+
});
|
|
233
|
+
break;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
case "accent-token": {
|
|
237
|
+
// Used internally by accent symbols.
|
|
238
|
+
break;
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
case "atom": {
|
|
242
|
+
const {text} = tree;
|
|
243
|
+
switch (tree.family) {
|
|
244
|
+
case "bin": {
|
|
245
|
+
buildString(text, "bin", a11yStrings);
|
|
246
|
+
break;
|
|
247
|
+
}
|
|
248
|
+
case "close": {
|
|
249
|
+
buildString(text, "close", a11yStrings);
|
|
250
|
+
break;
|
|
251
|
+
}
|
|
252
|
+
// TODO(kevinb): figure out what should be done for inner
|
|
253
|
+
case "inner": {
|
|
254
|
+
buildString(tree.text, "inner", a11yStrings);
|
|
255
|
+
break;
|
|
256
|
+
}
|
|
257
|
+
case "open": {
|
|
258
|
+
buildString(text, "open", a11yStrings);
|
|
259
|
+
break;
|
|
260
|
+
}
|
|
261
|
+
case "punct": {
|
|
262
|
+
buildString(text, "punct", a11yStrings);
|
|
263
|
+
break;
|
|
264
|
+
}
|
|
265
|
+
case "rel": {
|
|
266
|
+
buildString(text, "rel", a11yStrings);
|
|
267
|
+
break;
|
|
268
|
+
}
|
|
269
|
+
default: {
|
|
270
|
+
(tree.family: empty);
|
|
271
|
+
throw new Error(`"${tree.family}" is not a valid atom type`);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
break;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
case "color": {
|
|
278
|
+
const color = tree.color.replace(/katex-/, "");
|
|
279
|
+
|
|
280
|
+
buildRegion(a11yStrings, (regionStrings) => {
|
|
281
|
+
regionStrings.push("start color " + color);
|
|
282
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
283
|
+
regionStrings.push("end color " + color);
|
|
284
|
+
});
|
|
285
|
+
break;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
case "color-token": {
|
|
289
|
+
// Used by \color, \colorbox, and \fcolorbox but not directly rendered.
|
|
290
|
+
// It's a leaf node and has no children so just break.
|
|
291
|
+
break;
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
case "delimsizing": {
|
|
295
|
+
if (tree.delim && tree.delim !== ".") {
|
|
296
|
+
buildString(tree.delim, "normal", a11yStrings);
|
|
297
|
+
}
|
|
298
|
+
break;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
case "genfrac": {
|
|
302
|
+
buildRegion(a11yStrings, (regionStrings) => {
|
|
303
|
+
// genfrac can have unbalanced delimiters
|
|
304
|
+
const {leftDelim, rightDelim} = tree;
|
|
305
|
+
|
|
306
|
+
// NOTE: Not sure if this is a safe assumption
|
|
307
|
+
// hasBarLine true -> fraction, false -> binomial
|
|
308
|
+
if (tree.hasBarLine) {
|
|
309
|
+
regionStrings.push("start fraction");
|
|
310
|
+
leftDelim && buildString(leftDelim, "open", regionStrings);
|
|
311
|
+
buildA11yStrings(tree.numer, regionStrings, atomType);
|
|
312
|
+
regionStrings.push("divided by");
|
|
313
|
+
buildA11yStrings(tree.denom, regionStrings, atomType);
|
|
314
|
+
rightDelim && buildString(rightDelim, "close", regionStrings);
|
|
315
|
+
regionStrings.push("end fraction");
|
|
316
|
+
} else {
|
|
317
|
+
regionStrings.push("start binomial");
|
|
318
|
+
leftDelim && buildString(leftDelim, "open", regionStrings);
|
|
319
|
+
buildA11yStrings(tree.numer, regionStrings, atomType);
|
|
320
|
+
regionStrings.push("over");
|
|
321
|
+
buildA11yStrings(tree.denom, regionStrings, atomType);
|
|
322
|
+
rightDelim && buildString(rightDelim, "close", regionStrings);
|
|
323
|
+
regionStrings.push("end binomial");
|
|
324
|
+
}
|
|
325
|
+
});
|
|
326
|
+
break;
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
case "kern": {
|
|
330
|
+
// No op: we don't attempt to present kerning information
|
|
331
|
+
// to the screen reader.
|
|
332
|
+
break;
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
case "leftright": {
|
|
336
|
+
buildRegion(a11yStrings, (regionStrings) => {
|
|
337
|
+
buildString(tree.left, "open", regionStrings);
|
|
338
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
339
|
+
buildString(tree.right, "close", regionStrings);
|
|
340
|
+
});
|
|
341
|
+
break;
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
case "leftright-right": {
|
|
345
|
+
// TODO: double check that this is a no-op
|
|
346
|
+
break;
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
case "lap": {
|
|
350
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
351
|
+
break;
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
case "mathord": {
|
|
355
|
+
buildString(tree.text, "normal", a11yStrings);
|
|
356
|
+
break;
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
case "op": {
|
|
360
|
+
const {body, name} = tree;
|
|
361
|
+
if (body) {
|
|
362
|
+
buildA11yStrings(body, a11yStrings, atomType);
|
|
363
|
+
} else if (name) {
|
|
364
|
+
buildString(name, "normal", a11yStrings);
|
|
365
|
+
}
|
|
366
|
+
break;
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
case "op-token": {
|
|
370
|
+
// Used internally by operator symbols.
|
|
371
|
+
buildString(tree.text, atomType, a11yStrings);
|
|
372
|
+
break;
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
case "ordgroup": {
|
|
376
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
377
|
+
break;
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
case "overline": {
|
|
381
|
+
buildRegion(a11yStrings, function(a11yStrings) {
|
|
382
|
+
a11yStrings.push("start overline");
|
|
383
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
384
|
+
a11yStrings.push("end overline");
|
|
385
|
+
});
|
|
386
|
+
break;
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
case "phantom": {
|
|
390
|
+
a11yStrings.push("empty space");
|
|
391
|
+
break;
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
case "raisebox": {
|
|
395
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
396
|
+
break;
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
case "rule": {
|
|
400
|
+
a11yStrings.push("rectangle");
|
|
401
|
+
break;
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
case "sizing": {
|
|
405
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
406
|
+
break;
|
|
407
|
+
}
|
|
408
|
+
|
|
409
|
+
case "spacing": {
|
|
410
|
+
a11yStrings.push("space");
|
|
411
|
+
break;
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
case "styling": {
|
|
415
|
+
// We ignore the styling and just pass through the contents
|
|
416
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
417
|
+
break;
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
case "sqrt": {
|
|
421
|
+
buildRegion(a11yStrings, (regionStrings) => {
|
|
422
|
+
const {body, index} = tree;
|
|
423
|
+
if (index) {
|
|
424
|
+
const indexString = flatten(
|
|
425
|
+
buildA11yStrings(index, [], atomType)).join(",");
|
|
426
|
+
if (indexString === "3") {
|
|
427
|
+
regionStrings.push("cube root of");
|
|
428
|
+
buildA11yStrings(body, regionStrings, atomType);
|
|
429
|
+
regionStrings.push("end cube root");
|
|
430
|
+
return;
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
regionStrings.push("root");
|
|
434
|
+
regionStrings.push("start index");
|
|
435
|
+
buildA11yStrings(index, regionStrings, atomType);
|
|
436
|
+
regionStrings.push("end index");
|
|
437
|
+
return;
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
regionStrings.push("square root of");
|
|
441
|
+
buildA11yStrings(body, regionStrings, atomType);
|
|
442
|
+
regionStrings.push("end square root");
|
|
443
|
+
});
|
|
444
|
+
break;
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
case "supsub": {
|
|
448
|
+
const {base, sub, sup} = tree;
|
|
449
|
+
let isLog = false;
|
|
450
|
+
|
|
451
|
+
if (base) {
|
|
452
|
+
buildA11yStrings(base, a11yStrings, atomType);
|
|
453
|
+
isLog = base.type === "op" && base.name === "\\log";
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
if (sub) {
|
|
457
|
+
const regionName = isLog ? "base" : "subscript";
|
|
458
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
459
|
+
regionStrings.push(`start ${regionName}`);
|
|
460
|
+
buildA11yStrings(sub, regionStrings, atomType);
|
|
461
|
+
regionStrings.push(`end ${regionName}`);
|
|
462
|
+
});
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
if (sup) {
|
|
466
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
467
|
+
const supString = flatten(
|
|
468
|
+
buildA11yStrings(sup, [], atomType)).join(",");
|
|
469
|
+
|
|
470
|
+
if (supString in powerMap) {
|
|
471
|
+
regionStrings.push(powerMap[supString]);
|
|
472
|
+
return;
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
regionStrings.push("start superscript");
|
|
476
|
+
buildA11yStrings(sup, regionStrings, atomType);
|
|
477
|
+
regionStrings.push("end superscript");
|
|
478
|
+
});
|
|
479
|
+
}
|
|
480
|
+
break;
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
case "text": {
|
|
484
|
+
// TODO: handle other fonts
|
|
485
|
+
if (tree.font === "\\textbf") {
|
|
486
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
487
|
+
regionStrings.push("start bold text");
|
|
488
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
489
|
+
regionStrings.push("end bold text");
|
|
490
|
+
});
|
|
491
|
+
break;
|
|
492
|
+
}
|
|
493
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
494
|
+
regionStrings.push("start text");
|
|
495
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
496
|
+
regionStrings.push("end text");
|
|
497
|
+
});
|
|
498
|
+
break;
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
case "textord": {
|
|
502
|
+
buildString(tree.text, atomType, a11yStrings);
|
|
503
|
+
break;
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
case "smash": {
|
|
507
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
508
|
+
break;
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
case "enclose": {
|
|
512
|
+
// TODO: create a map for these.
|
|
513
|
+
// TODO: differentiate between a body with a single atom, e.g.
|
|
514
|
+
// "cancel a" instead of "start cancel, a, end cancel"
|
|
515
|
+
if (/cancel/.test(tree.label)) {
|
|
516
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
517
|
+
regionStrings.push("start cancel");
|
|
518
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
519
|
+
regionStrings.push("end cancel");
|
|
520
|
+
});
|
|
521
|
+
break;
|
|
522
|
+
} else if (/box/.test(tree.label)) {
|
|
523
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
524
|
+
regionStrings.push("start box");
|
|
525
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
526
|
+
regionStrings.push("end box");
|
|
527
|
+
});
|
|
528
|
+
break;
|
|
529
|
+
} else if (/sout/.test(tree.label)) {
|
|
530
|
+
buildRegion(a11yStrings, function(regionStrings) {
|
|
531
|
+
regionStrings.push("start strikeout");
|
|
532
|
+
buildA11yStrings(tree.body, regionStrings, atomType);
|
|
533
|
+
regionStrings.push("end strikeout");
|
|
534
|
+
});
|
|
535
|
+
break;
|
|
536
|
+
}
|
|
537
|
+
throw new Error(
|
|
538
|
+
`KaTeX-a11y: enclose node with ${tree.label} not supported yet`);
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
case "vphantom": {
|
|
542
|
+
throw new Error("KaTeX-a11y: vphantom not implemented yet");
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
case "hphantom": {
|
|
546
|
+
throw new Error("KaTeX-a11y: hphantom not implemented yet");
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
case "operatorname": {
|
|
550
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
551
|
+
break;
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
case "array": {
|
|
555
|
+
throw new Error("KaTeX-a11y: array not implemented yet");
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
case "raw": {
|
|
559
|
+
throw new Error("KaTeX-a11y: raw not implemented yet");
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
case "size": {
|
|
563
|
+
// Although there are nodes of type "size" in the parse tree, they have
|
|
564
|
+
// no semantic meaning and should be ignored.
|
|
565
|
+
break;
|
|
566
|
+
}
|
|
567
|
+
|
|
568
|
+
case "url": {
|
|
569
|
+
throw new Error("KaTeX-a11y: url not implemented yet");
|
|
570
|
+
}
|
|
571
|
+
|
|
572
|
+
case "tag": {
|
|
573
|
+
throw new Error("KaTeX-a11y: tag not implemented yet");
|
|
574
|
+
}
|
|
575
|
+
|
|
576
|
+
case "verb": {
|
|
577
|
+
buildString(`start verbatim`, "normal", a11yStrings);
|
|
578
|
+
buildString(tree.body, "normal", a11yStrings);
|
|
579
|
+
buildString(`end verbatim`, "normal", a11yStrings);
|
|
580
|
+
break;
|
|
581
|
+
}
|
|
582
|
+
|
|
583
|
+
case "environment": {
|
|
584
|
+
throw new Error("KaTeX-a11y: environment not implemented yet");
|
|
585
|
+
}
|
|
586
|
+
|
|
587
|
+
case "horizBrace": {
|
|
588
|
+
buildString(`start ${tree.label.slice(1)}`, "normal", a11yStrings);
|
|
589
|
+
buildA11yStrings(tree.base, a11yStrings, atomType);
|
|
590
|
+
buildString(`end ${tree.label.slice(1)}`, "normal", a11yStrings);
|
|
591
|
+
break;
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
case "infix": {
|
|
595
|
+
// All infix nodes are replace with other nodes.
|
|
596
|
+
break;
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
case "includegraphics": {
|
|
600
|
+
throw new Error("KaTeX-a11y: includegraphics not implemented yet");
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
case "font": {
|
|
604
|
+
// TODO: callout the start/end of specific fonts
|
|
605
|
+
// TODO: map \BBb{N} to "the naturals" or something like that
|
|
606
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
607
|
+
break;
|
|
608
|
+
}
|
|
609
|
+
|
|
610
|
+
case "href": {
|
|
611
|
+
throw new Error("KaTeX-a11y: href not implemented yet");
|
|
612
|
+
}
|
|
613
|
+
|
|
614
|
+
case "cr": {
|
|
615
|
+
// This is used by environments.
|
|
616
|
+
throw new Error("KaTeX-a11y: cr not implemented yet");
|
|
617
|
+
}
|
|
618
|
+
|
|
619
|
+
case "underline": {
|
|
620
|
+
buildRegion(a11yStrings, function(a11yStrings) {
|
|
621
|
+
a11yStrings.push("start underline");
|
|
622
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
623
|
+
a11yStrings.push("end underline");
|
|
624
|
+
});
|
|
625
|
+
break;
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
case "xArrow": {
|
|
629
|
+
throw new Error("KaTeX-a11y: xArrow not implemented yet");
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
case "mclass": {
|
|
633
|
+
// \neq and \ne are macros so we let "htmlmathml" render the mathmal
|
|
634
|
+
// side of things and extract the text from that.
|
|
635
|
+
const atomType = tree.mclass.slice(1);
|
|
636
|
+
// $FlowFixMe: drop the leading "m" from the values in mclass
|
|
637
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
638
|
+
break;
|
|
639
|
+
}
|
|
640
|
+
|
|
641
|
+
case "mathchoice": {
|
|
642
|
+
// TODO: track which which style we're using, e.g. dispaly, text, etc.
|
|
643
|
+
// default to text style if even that may not be the correct style
|
|
644
|
+
buildA11yStrings(tree.text, a11yStrings, atomType);
|
|
645
|
+
break;
|
|
646
|
+
}
|
|
647
|
+
|
|
648
|
+
case "htmlmathml": {
|
|
649
|
+
buildA11yStrings(tree.mathml, a11yStrings, atomType);
|
|
650
|
+
break;
|
|
651
|
+
}
|
|
652
|
+
|
|
653
|
+
case "middle": {
|
|
654
|
+
buildString(tree.delim, atomType, a11yStrings);
|
|
655
|
+
break;
|
|
656
|
+
}
|
|
657
|
+
|
|
658
|
+
case "internal": {
|
|
659
|
+
// internal nodes are never included in the parse tree
|
|
660
|
+
break;
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
case "html": {
|
|
664
|
+
buildA11yStrings(tree.body, a11yStrings, atomType);
|
|
665
|
+
break;
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
default:
|
|
669
|
+
(tree.type: empty);
|
|
670
|
+
throw new Error("KaTeX a11y un-recognized type: " + tree.type);
|
|
671
|
+
}
|
|
672
|
+
};
|
|
673
|
+
|
|
674
|
+
const buildA11yStrings = (
|
|
675
|
+
tree: AnyParseNode | AnyParseNode[],
|
|
676
|
+
a11yStrings: NestedArray<string> = [],
|
|
677
|
+
atomType: Atom | "normal",
|
|
678
|
+
) => {
|
|
679
|
+
if (tree instanceof Array) {
|
|
680
|
+
for (let i = 0; i < tree.length; i++) {
|
|
681
|
+
buildA11yStrings(tree[i], a11yStrings, atomType);
|
|
682
|
+
}
|
|
683
|
+
} else {
|
|
684
|
+
handleObject(tree, a11yStrings, atomType);
|
|
685
|
+
}
|
|
686
|
+
|
|
687
|
+
return a11yStrings;
|
|
688
|
+
};
|
|
689
|
+
|
|
690
|
+
|
|
691
|
+
const flatten = function(array) {
|
|
692
|
+
let result = [];
|
|
693
|
+
|
|
694
|
+
array.forEach(function(item) {
|
|
695
|
+
if (item instanceof Array) {
|
|
696
|
+
result = result.concat(flatten(item));
|
|
697
|
+
} else {
|
|
698
|
+
result.push(item);
|
|
699
|
+
}
|
|
700
|
+
});
|
|
701
|
+
|
|
702
|
+
return result;
|
|
703
|
+
};
|
|
704
|
+
|
|
705
|
+
const renderA11yString = function(text: string, settings?: SettingsOptions) {
|
|
706
|
+
|
|
707
|
+
const tree = katex.__parse(text, settings);
|
|
708
|
+
const a11yStrings = buildA11yStrings(tree, [], "normal");
|
|
709
|
+
return flatten(a11yStrings).join(", ");
|
|
710
|
+
};
|
|
711
|
+
|
|
712
|
+
export default renderA11yString;
|