mirror of
				https://github.com/kovidgoyal/calibre.git
				synced 2025-10-31 10:37:00 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			2276 lines
		
	
	
		
			87 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			2276 lines
		
	
	
		
			87 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* -*- Mode: Javascript; indent-tabs-mode:nil; js-indent-level: 2 -*- */
 | |
| /* vim: set ts=2 et sw=2 tw=80: */
 | |
| 
 | |
| /*************************************************************
 | |
|  *
 | |
|  *  MathJax/jax/input/TeX/jax.js
 | |
|  *  
 | |
|  *  Implements the TeX InputJax that reads mathematics in
 | |
|  *  TeX and LaTeX format and converts it to the MML ElementJax
 | |
|  *  internal format.
 | |
|  *
 | |
|  *  ---------------------------------------------------------------------
 | |
|  *  
 | |
|  *  Copyright (c) 2009-2015 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.
 | |
|  */
 | |
| 
 | |
| (function (TEX,HUB,AJAX) {
 | |
|   var MML, NBSP = "\u00A0"; 
 | |
|   
 | |
|   var _ = function (id) {
 | |
|     return MathJax.Localization._.apply(MathJax.Localization,
 | |
|       [["TeX", id]].concat([].slice.call(arguments,1)));
 | |
|   };
 | |
| 
 | |
|   var STACK = MathJax.Object.Subclass({
 | |
|     Init: function (env,inner) {
 | |
|       this.global = {isInner: inner};
 | |
|       this.data = [STACKITEM.start(this.global)];
 | |
|       if (env) {this.data[0].env = env}
 | |
|       this.env = this.data[0].env;
 | |
|     },
 | |
|     Push: function () {
 | |
|       var i, m, item, top;
 | |
|       for (i = 0, m = arguments.length; i < m; i++) {
 | |
|         item = arguments[i]; if (!item) continue;
 | |
|         if (item instanceof MML.mbase) {item = STACKITEM.mml(item)}
 | |
|         item.global = this.global;
 | |
|         top = (this.data.length ? this.Top().checkItem(item) : true);
 | |
|         if (top instanceof Array) {this.Pop(); this.Push.apply(this,top)}
 | |
|         else if (top instanceof STACKITEM) {this.Pop(); this.Push(top)}
 | |
|         else if (top) {
 | |
|           this.data.push(item);
 | |
|           if (item.env) {
 | |
|             for (var id in this.env)
 | |
|               {if (this.env.hasOwnProperty(id)) {item.env[id] = this.env[id]}}
 | |
|             this.env = item.env;
 | |
|           } else {item.env = this.env}
 | |
|         }
 | |
|       }
 | |
|     },
 | |
|     Pop: function () {
 | |
|       var item = this.data.pop(); if (!item.isOpen) {delete item.env}
 | |
|       this.env = (this.data.length ? this.Top().env : {});
 | |
|       return item;
 | |
|     },
 | |
|     Top: function (n) {
 | |
|       if (n == null) {n = 1}
 | |
|       if (this.data.length < n) {return null}
 | |
|       return this.data[this.data.length-n];
 | |
|     },
 | |
|     Prev: function (noPop) {
 | |
|       var top = this.Top();
 | |
|       if (noPop) {return top.data[top.data.length-1]}
 | |
|             else {return top.Pop()}
 | |
|     },
 | |
|     toString: function () {return "stack[\n  "+this.data.join("\n  ")+"\n]"}
 | |
|   });
 | |
|   
 | |
|   var STACKITEM = STACK.Item = MathJax.Object.Subclass({
 | |
|     type: "base",
 | |
|     endError:   /*_()*/ ["ExtraOpenMissingClose","Extra open brace or missing close brace"],
 | |
|     closeError: /*_()*/ ["ExtraCloseMissingOpen","Extra close brace or missing open brace"],
 | |
|     rightError: /*_()*/ ["MissingLeftExtraRight","Missing \\left or extra \\right"],
 | |
|     Init: function () {
 | |
|       if (this.isOpen) {this.env = {}}
 | |
|       this.data = [];
 | |
|       this.Push.apply(this,arguments);
 | |
|     },
 | |
|     Push: function () {this.data.push.apply(this.data,arguments)},
 | |
|     Pop: function () {return this.data.pop()},
 | |
|     mmlData: function (inferred,forceRow) {
 | |
|       if (inferred == null) {inferred = true}
 | |
|       if (this.data.length === 1 && !forceRow) {return this.data[0]}
 | |
|       return MML.mrow.apply(MML,this.data).With((inferred ? {inferred: true}: {}));
 | |
|     },
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "over" && this.isOpen) {item.num = this.mmlData(false); this.data = []}
 | |
|       if (item.type === "cell" && this.isOpen) {
 | |
|         if (item.linebreak) {return false}
 | |
|         TEX.Error(["Misplaced","Misplaced %1",item.name]);
 | |
|       }
 | |
|       if (item.isClose && this[item.type+"Error"]) {TEX.Error(this[item.type+"Error"])}
 | |
|       if (!item.isNotStack) {return true}
 | |
|       this.Push(item.data[0]); return false;
 | |
|     },
 | |
|     With: function (def) {
 | |
|       for (var id in def) {if (def.hasOwnProperty(id)) {this[id] = def[id]}}
 | |
|       return this;
 | |
|     },
 | |
|     toString: function () {return this.type+"["+this.data.join("; ")+"]"}
 | |
|   });
 | |
| 
 | |
|   STACKITEM.start = STACKITEM.Subclass({
 | |
|     type: "start", isOpen: true,
 | |
|     Init: function (global) {
 | |
|       this.SUPER(arguments).Init.call(this);
 | |
|       this.global = global;
 | |
|     },
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "stop") {return STACKITEM.mml(this.mmlData())}
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   STACKITEM.stop = STACKITEM.Subclass({
 | |
|     type: "stop", isClose: true
 | |
|   });
 | |
| 
 | |
|   STACKITEM.open = STACKITEM.Subclass({
 | |
|     type: "open", isOpen: true,
 | |
|     stopError: /*_()*/ ["ExtraOpenMissingClose","Extra open brace or missing close brace"],
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "close") {
 | |
|         var mml = this.mmlData();
 | |
|         return STACKITEM.mml(MML.TeXAtom(mml)); // TeXAtom make it an ORD to prevent spacing (FIXME: should be another way)
 | |
|       }
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   STACKITEM.close = STACKITEM.Subclass({
 | |
|     type: "close", isClose: true
 | |
|   });
 | |
| 
 | |
|   STACKITEM.prime = STACKITEM.Subclass({
 | |
|     type: "prime",
 | |
|     checkItem: function (item) {
 | |
|       if (this.data[0].type !== "msubsup") 
 | |
|         {return [MML.msup(this.data[0],this.data[1]),item]}
 | |
|       this.data[0].SetData(this.data[0].sup,this.data[1]);
 | |
|       return [this.data[0],item];
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.subsup = STACKITEM.Subclass({
 | |
|     type: "subsup",
 | |
|     stopError: /*_()*/ ["MissingScript","Missing superscript or subscript argument"],
 | |
|     supError:  /*_()*/ ["MissingOpenForSup","Missing open brace for superscript"],
 | |
|     subError:  /*_()*/ ["MissingOpenForSub","Missing open brace for subscript"],
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "open" || item.type === "left") {return true}
 | |
|       if (item.type === "mml") {
 | |
|         if (this.primes) {
 | |
|           if (this.position !== 2) {this.data[0].SetData(2,this.primes)}
 | |
|             else {item.data[0] = MML.mrow(this.primes.With({variantForm:true}),item.data[0])}
 | |
|         }
 | |
|         this.data[0].SetData(this.position,item.data[0]);
 | |
|         if (this.movesupsub != null) {this.data[0].movesupsub = this.movesupsub}
 | |
|         return STACKITEM.mml(this.data[0]);
 | |
|       }
 | |
|       if (this.SUPER(arguments).checkItem.call(this,item))
 | |
|         {TEX.Error(this[["","subError","supError"][this.position]])}
 | |
|     },
 | |
|     Pop: function () {}
 | |
|   });
 | |
| 
 | |
|   STACKITEM.over = STACKITEM.Subclass({
 | |
|     type: "over", isClose: true, name: "\\over",
 | |
|     checkItem: function (item,stack) {
 | |
|       if (item.type === "over")
 | |
|         {TEX.Error(["AmbiguousUseOf","Ambiguous use of %1",item.name])}
 | |
|       if (item.isClose) {
 | |
|         var mml = MML.mfrac(this.num,this.mmlData(false));
 | |
|         if (this.thickness != null) {mml.linethickness = this.thickness}
 | |
|         if (this.open || this.close) {
 | |
|           mml.texWithDelims = true;
 | |
|           mml = TEX.fixedFence(this.open,mml,this.close);
 | |
|         }
 | |
|         return [STACKITEM.mml(mml), item];
 | |
|       }
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     },
 | |
|     toString: function () {return "over["+this.num+" / "+this.data.join("; ")+"]"}
 | |
|   });
 | |
| 
 | |
|   STACKITEM.left = STACKITEM.Subclass({
 | |
|     type: "left", isOpen: true, delim: '(',
 | |
|     stopError: /*_()*/ ["ExtraLeftMissingRight", "Extra \\left or missing \\right"],
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "right")
 | |
|         {return STACKITEM.mml(TEX.fenced(this.delim,this.mmlData(),item.delim))}
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   STACKITEM.right = STACKITEM.Subclass({
 | |
|     type: "right", isClose: true, delim: ')'
 | |
|   });
 | |
| 
 | |
|   STACKITEM.begin = STACKITEM.Subclass({
 | |
|     type: "begin", isOpen: true,
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "end") {
 | |
|         if (item.name !== this.name)
 | |
|           {TEX.Error(["EnvBadEnd","\\begin{%1} ended with \\end{%2}",this.name,item.name])}
 | |
|         if (!this.end) {return STACKITEM.mml(this.mmlData())}
 | |
|         return this.parse[this.end].call(this.parse,this,this.data);
 | |
|       }
 | |
|       if (item.type === "stop")
 | |
|         {TEX.Error(["EnvMissingEnd","Missing \\end{%1}",this.name])}
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.end = STACKITEM.Subclass({
 | |
|     type: "end", isClose: true
 | |
|   });
 | |
| 
 | |
|   STACKITEM.style = STACKITEM.Subclass({
 | |
|     type: "style",
 | |
|     checkItem: function (item) {
 | |
|       if (!item.isClose) {return this.SUPER(arguments).checkItem.call(this,item)}
 | |
|       var mml = MML.mstyle.apply(MML,this.data).With(this.styles);
 | |
|       return [STACKITEM.mml(mml),item];
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.position = STACKITEM.Subclass({
 | |
|     type: "position",
 | |
|     checkItem: function (item) {
 | |
|       if (item.isClose) {TEX.Error(["MissingBoxFor","Missing box for %1",this.name])}
 | |
|       if (item.isNotStack) {
 | |
|         var mml = item.mmlData();
 | |
|         switch (this.move) {
 | |
|          case 'vertical':
 | |
|           mml = MML.mpadded(mml).With({height: this.dh, depth: this.dd, voffset: this.dh});
 | |
|           return [STACKITEM.mml(mml)];
 | |
|          case 'horizontal':
 | |
|           return [STACKITEM.mml(this.left),item,STACKITEM.mml(this.right)];
 | |
|         }
 | |
|       }
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.array = STACKITEM.Subclass({
 | |
|     type: "array", isOpen: true, arraydef: {},
 | |
|     Init: function () {
 | |
|       this.table = []; this.row = []; this.env = {}; this.frame = []; this.hfill = [];
 | |
|       this.SUPER(arguments).Init.apply(this,arguments);
 | |
|     },
 | |
|     checkItem: function (item) {
 | |
|       if (item.isClose && item.type !== "over") {
 | |
|         if (item.isEntry) {this.EndEntry(); this.clearEnv(); return false}
 | |
|         if (item.isCR)    {this.EndEntry(); this.EndRow(); this.clearEnv(); return false}
 | |
|         this.EndTable(); this.clearEnv();
 | |
|         var scriptlevel = this.arraydef.scriptlevel; delete this.arraydef.scriptlevel;
 | |
|         var mml = MML.mtable.apply(MML,this.table).With(this.arraydef);
 | |
|         if (this.frame.length === 4) {
 | |
|           mml.frame = (this.frame.dashed ? "dashed" : "solid");
 | |
|         } else if (this.frame.length) {
 | |
|           mml.hasFrame = true;
 | |
|           if (this.arraydef.rowlines) {this.arraydef.rowlines = this.arraydef.rowlines.replace(/none( none)+$/,"none")}
 | |
|           mml = MML.menclose(mml).With({notation: this.frame.join(" "), isFrame: true});
 | |
|           if ((this.arraydef.columnlines||"none") != "none" ||
 | |
|               (this.arraydef.rowlines||"none") != "none") {mml.padding = 0} // HTML-CSS jax implements this
 | |
|         }
 | |
|         if (scriptlevel) {mml = MML.mstyle(mml).With({scriptlevel: scriptlevel})}
 | |
|         if (this.open || this.close) {mml = TEX.fenced(this.open,mml,this.close)}
 | |
|         mml = STACKITEM.mml(mml);
 | |
|         if (this.requireClose) {
 | |
|           if (item.type === 'close') {return mml}
 | |
|           TEX.Error(["MissingCloseBrace","Missing close brace"]);
 | |
|         }
 | |
|         return [mml,item];
 | |
|       }
 | |
|       return this.SUPER(arguments).checkItem.call(this,item);
 | |
|     },
 | |
|     EndEntry: function () {
 | |
|       var mtd = MML.mtd.apply(MML,this.data);
 | |
|       if (this.hfill.length) {
 | |
|         if (this.hfill[0] === 0) mtd.columnalign = "right";
 | |
|         if (this.hfill[this.hfill.length-1] === this.data.length)
 | |
|           mtd.columnalign = (mtd.columnalign ? "center" : "left");
 | |
|       }
 | |
|       this.row.push(mtd); this.data = []; this.hfill = [];
 | |
|     },
 | |
|     EndRow:   function () {
 | |
|       var mtr = MML.mtr;
 | |
|       if (this.isNumbered && this.row.length === 3) {
 | |
|         this.row.unshift(this.row.pop());  // move equation number to first position
 | |
|         mtr = MML.mlabeledtr;
 | |
|       }
 | |
|       this.table.push(mtr.apply(MML,this.row)); this.row = [];
 | |
|     },
 | |
|     EndTable: function () {
 | |
|       if (this.data.length || this.row.length) {this.EndEntry(); this.EndRow()}
 | |
|       this.checkLines();
 | |
|     },
 | |
|     checkLines: function () {
 | |
|       if (this.arraydef.rowlines) {
 | |
|         var lines = this.arraydef.rowlines.split(/ /);
 | |
|         if (lines.length === this.table.length) {
 | |
|           this.frame.push("bottom"); lines.pop();
 | |
|           this.arraydef.rowlines = lines.join(' ');
 | |
|         } else if (lines.length < this.table.length-1) {
 | |
|           this.arraydef.rowlines += " none";
 | |
|         }
 | |
|       }
 | |
|       if (this.rowspacing) {
 | |
|         var rows = this.arraydef.rowspacing.split(/ /);
 | |
|         while (rows.length < this.table.length) {rows.push(this.rowspacing+"em")}
 | |
|         this.arraydef.rowspacing = rows.join(' ');
 | |
|       }
 | |
|     },
 | |
|     clearEnv: function () {
 | |
|       for (var id in this.env) {if (this.env.hasOwnProperty(id)) {delete this.env[id]}}
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.cell = STACKITEM.Subclass({
 | |
|     type: "cell", isClose: true
 | |
|   });
 | |
| 
 | |
|   STACKITEM.mml = STACKITEM.Subclass({
 | |
|     type: "mml", isNotStack: true,
 | |
|     Add: function () {this.data.push.apply(this.data,arguments); return this}
 | |
|   });
 | |
|   
 | |
|   STACKITEM.fn = STACKITEM.Subclass({
 | |
|     type: "fn",
 | |
|     checkItem: function (item) {
 | |
|       if (this.data[0]) {
 | |
|         if (item.isOpen) {return true}
 | |
|         if (item.type !== "fn") {
 | |
|           if (item.type !== "mml" || !item.data[0]) {return [this.data[0],item]}
 | |
|           if (item.data[0].isa(MML.mspace)) {return [this.data[0],item]}
 | |
|           var mml = item.data[0]; if (mml.isEmbellished()) {mml = mml.CoreMO()}
 | |
|           if ([0,0,1,1,0,1,1,0,0,0][mml.Get("texClass")]) {return [this.data[0],item]}
 | |
|         }
 | |
|         return [this.data[0],MML.mo(MML.entity("#x2061")).With({texClass:MML.TEXCLASS.NONE}),item];
 | |
|       }
 | |
|       return this.SUPER(arguments).checkItem.apply(this,arguments);
 | |
|     }
 | |
|   });
 | |
|   
 | |
|   STACKITEM.not = STACKITEM.Subclass({
 | |
|     type: "not",
 | |
|     checkItem: function (item) {
 | |
|       var mml, c;
 | |
|       if (item.type === "open" || item.type === "left") {return true}
 | |
|       if (item.type === "mml" && item.data[0].type.match(/^(mo|mi|mtext)$/)) {
 | |
|         mml = item.data[0], c = mml.data.join("");
 | |
|         if (c.length === 1 && !mml.movesupsub) {
 | |
|           c = STACKITEM.not.remap[c.charCodeAt(0)];
 | |
|           if (c) {mml.SetData(0,MML.chars(String.fromCharCode(c)))}
 | |
|             else {mml.Append(MML.chars("\u0338"))}
 | |
|           return item;
 | |
|         }
 | |
|       }
 | |
|       //  \mathrel{\rlap{\notChar}}
 | |
|       mml = MML.mpadded(MML.mtext("\u29F8")).With({width:0});
 | |
|       mml = MML.TeXAtom(mml).With({texClass:MML.TEXCLASS.REL});
 | |
|       return [mml,item];
 | |
|     }
 | |
|   });
 | |
|   STACKITEM.not.remap = {
 | |
|     0x2190:0x219A, 0x2192:0x219B, 0x2194:0x21AE,
 | |
|     0x21D0:0x21CD, 0x21D2:0x21CF, 0x21D4:0x21CE,
 | |
|     0x2208:0x2209, 0x220B:0x220C, 0x2223:0x2224, 0x2225:0x2226,
 | |
|     0x223C:0x2241, 0x007E:0x2241, 0x2243:0x2244, 0x2245:0x2247,
 | |
|     0x2248:0x2249, 0x224D:0x226D, 0x003D:0x2260, 0x2261:0x2262,
 | |
|     0x003C:0x226E, 0x003E:0x226F, 0x2264:0x2270, 0x2265:0x2271,
 | |
|     0x2272:0x2274, 0x2273:0x2275, 0x2276:0x2278, 0x2277:0x2279,
 | |
|     0x227A:0x2280, 0x227B:0x2281, 0x2282:0x2284, 0x2283:0x2285,
 | |
|     0x2286:0x2288, 0x2287:0x2289, 0x22A2:0x22AC, 0x22A8:0x22AD,
 | |
|     0x22A9:0x22AE, 0x22AB:0x22AF, 0x227C:0x22E0, 0x227D:0x22E1,
 | |
|     0x2291:0x22E2, 0x2292:0x22E3, 0x22B2:0x22EA, 0x22B3:0x22EB,
 | |
|     0x22B4:0x22EC, 0x22B5:0x22ED, 0x2203:0x2204
 | |
|   };
 | |
|   
 | |
|   STACKITEM.dots = STACKITEM.Subclass({
 | |
|     type: "dots",
 | |
|     checkItem: function (item) {
 | |
|       if (item.type === "open" || item.type === "left") {return true}
 | |
|       var dots = this.ldots;
 | |
|       if (item.type === "mml" && item.data[0].isEmbellished()) {
 | |
|         var tclass = item.data[0].CoreMO().Get("texClass");
 | |
|         if (tclass === MML.TEXCLASS.BIN || tclass === MML.TEXCLASS.REL) {dots = this.cdots}
 | |
|       }
 | |
|       return [dots,item];
 | |
|     }
 | |
|   });
 | |
|   
 | |
| 
 | |
|   var TEXDEF = {
 | |
|     //
 | |
|     //  Add new definitions without overriding user-defined ones
 | |
|     //
 | |
|     Add: function (src,dst,nouser) {
 | |
|       if (!dst) {dst = this}
 | |
|       for (var id in src) {if (src.hasOwnProperty(id)) {
 | |
|         if (typeof src[id] === 'object' && !(src[id] instanceof Array) &&
 | |
|            (typeof dst[id] === 'object' || typeof dst[id] === 'function')) 
 | |
|              {this.Add(src[id],dst[id],src[id],nouser)}
 | |
|           else if (!dst[id] || !dst[id].isUser || !nouser) {dst[id] = src[id]}
 | |
|       }}
 | |
|       return dst;
 | |
|     }
 | |
|   };
 | |
|   var STARTUP = function () {
 | |
|     MML = MathJax.ElementJax.mml;
 | |
|     HUB.Insert(TEXDEF,{
 | |
|   
 | |
|       // patterns for letters and numbers
 | |
|       letter:  /[a-z]/i,
 | |
|       digit:   /[0-9.]/,
 | |
|       number:  /^(?:[0-9]+(?:\{,\}[0-9]{3})*(?:\.[0-9]*)*|\.[0-9]+)/,
 | |
|     
 | |
|       special: {
 | |
|         '\\':  'ControlSequence',
 | |
|         '{':   'Open',
 | |
|         '}':   'Close',
 | |
|         '~':   'Tilde',
 | |
|         '^':   'Superscript',
 | |
|         '_':   'Subscript',
 | |
|         ' ':   'Space',
 | |
|         "\t":  'Space',
 | |
|         "\r":  'Space',
 | |
|         "\n":  'Space',
 | |
|         "'":   'Prime',
 | |
|         '%':   'Comment',
 | |
|         '&':   'Entry',
 | |
|         '#':   'Hash',
 | |
|         '\u00A0': 'Space',
 | |
|         '\u2019': 'Prime'
 | |
|       },
 | |
|       
 | |
|       remap: {
 | |
|         '-':   '2212',
 | |
|         '*':   '2217',
 | |
|         '`':   '2018'   // map ` to back quote
 | |
|       },
 | |
|     
 | |
|       mathchar0mi: {
 | |
| 	// Lower-case greek
 | |
| 	alpha:        '03B1',
 | |
| 	beta:         '03B2',
 | |
| 	gamma:        '03B3',
 | |
| 	delta:        '03B4',
 | |
| 	epsilon:      '03F5',
 | |
| 	zeta:         '03B6',
 | |
| 	eta:          '03B7',
 | |
| 	theta:        '03B8',
 | |
| 	iota:         '03B9',
 | |
| 	kappa:        '03BA',
 | |
| 	lambda:       '03BB',
 | |
| 	mu:           '03BC',
 | |
| 	nu:           '03BD',
 | |
| 	xi:           '03BE',
 | |
| 	omicron:      '03BF', // added for completeness
 | |
| 	pi:           '03C0',
 | |
| 	rho:          '03C1',
 | |
| 	sigma:        '03C3',
 | |
| 	tau:          '03C4',
 | |
| 	upsilon:      '03C5',
 | |
| 	phi:          '03D5',
 | |
| 	chi:          '03C7',
 | |
| 	psi:          '03C8',
 | |
| 	omega:        '03C9',
 | |
| 	varepsilon:   '03B5',
 | |
| 	vartheta:     '03D1',
 | |
| 	varpi:        '03D6',
 | |
| 	varrho:       '03F1',
 | |
| 	varsigma:     '03C2',
 | |
| 	varphi:       '03C6',
 | |
|         
 | |
|         // Ord symbols
 | |
|         S:            ['00A7',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         aleph:        ['2135',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         hbar:         ['210F',{variantForm:true}],
 | |
|         imath:        '0131',
 | |
|         jmath:        '0237',
 | |
|         ell:          '2113',
 | |
|         wp:           ['2118',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         Re:           ['211C',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         Im:           ['2111',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         partial:      ['2202',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         infty:        ['221E',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         prime:        ['2032',{mathvariant: MML.VARIANT.NORMAL, variantForm:true}],
 | |
|         emptyset:     ['2205',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         nabla:        ['2207',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         top:          ['22A4',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         bot:          ['22A5',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         angle:        ['2220',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         triangle:     ['25B3',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         backslash:    ['2216',{mathvariant: MML.VARIANT.NORMAL, variantForm:true}],
 | |
|         forall:       ['2200',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         exists:       ['2203',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         neg:          ['00AC',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         lnot:         ['00AC',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         flat:         ['266D',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         natural:      ['266E',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         sharp:        ['266F',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         clubsuit:     ['2663',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         diamondsuit:  ['2662',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         heartsuit:    ['2661',{mathvariant: MML.VARIANT.NORMAL}],
 | |
|         spadesuit:    ['2660',{mathvariant: MML.VARIANT.NORMAL}]
 | |
|       },
 | |
|         
 | |
|       mathchar0mo: {
 | |
|         surd:         '221A',
 | |
| 
 | |
|         // big ops
 | |
|         coprod:       ['2210',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigvee:       ['22C1',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigwedge:     ['22C0',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         biguplus:     ['2A04',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigcap:       ['22C2',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigcup:       ['22C3',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         'int':        ['222B',{texClass: MML.TEXCLASS.OP}],
 | |
|         intop:        ['222B',{texClass: MML.TEXCLASS.OP, movesupsub:true, movablelimits:true}],
 | |
|         iint:         ['222C',{texClass: MML.TEXCLASS.OP}],
 | |
|         iiint:        ['222D',{texClass: MML.TEXCLASS.OP}],
 | |
|         prod:         ['220F',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         sum:          ['2211',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigotimes:    ['2A02',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigoplus:     ['2A01',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         bigodot:      ['2A00',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         oint:         ['222E',{texClass: MML.TEXCLASS.OP}],
 | |
|         bigsqcup:     ['2A06',{texClass: MML.TEXCLASS.OP, movesupsub:true}],
 | |
|         smallint:     ['222B',{largeop:false}],
 | |
|         
 | |
|         // binary operations
 | |
|         triangleleft:      '25C3',
 | |
|         triangleright:     '25B9',
 | |
|         bigtriangleup:     '25B3',
 | |
|         bigtriangledown:   '25BD',
 | |
|         wedge:        '2227',
 | |
|         land:         '2227',
 | |
|         vee:          '2228',
 | |
|         lor:          '2228',
 | |
|         cap:          '2229',
 | |
|         cup:          '222A',
 | |
|         ddagger:      '2021',
 | |
|         dagger:       '2020',
 | |
|         sqcap:        '2293',
 | |
|         sqcup:        '2294',
 | |
|         uplus:        '228E',
 | |
|         amalg:        '2A3F',
 | |
|         diamond:      '22C4',
 | |
|         bullet:       '2219',
 | |
|         wr:           '2240',
 | |
|         div:          '00F7',
 | |
|         odot:         ['2299',{largeop: false}],
 | |
|         oslash:       ['2298',{largeop: false}],
 | |
|         otimes:       ['2297',{largeop: false}],
 | |
|         ominus:       ['2296',{largeop: false}],
 | |
|         oplus:        ['2295',{largeop: false}],
 | |
|         mp:           '2213',
 | |
|         pm:           '00B1',
 | |
|         circ:         '2218',
 | |
|         bigcirc:      '25EF',
 | |
|         setminus:     ['2216',{variantForm:true}],
 | |
|         cdot:         '22C5',
 | |
|         ast:          '2217',
 | |
|         times:        '00D7',
 | |
|         star:         '22C6',
 | |
|         
 | |
|         // Relations
 | |
|         propto:       '221D',
 | |
|         sqsubseteq:   '2291',
 | |
|         sqsupseteq:   '2292',
 | |
|         parallel:     '2225',
 | |
|         mid:          '2223',
 | |
|         dashv:        '22A3',
 | |
|         vdash:        '22A2',
 | |
|         leq:          '2264',
 | |
|         le:           '2264',
 | |
|         geq:          '2265',
 | |
|         ge:           '2265',
 | |
|         lt:           '003C',
 | |
|         gt:           '003E',
 | |
|         succ:         '227B',
 | |
|         prec:         '227A',
 | |
|         approx:       '2248',
 | |
|         succeq:       '2AB0',  // or '227C',
 | |
|         preceq:       '2AAF',  // or '227D',
 | |
|         supset:       '2283',
 | |
|         subset:       '2282',
 | |
|         supseteq:     '2287',
 | |
|         subseteq:     '2286',
 | |
|         'in':         '2208',
 | |
|         ni:           '220B',
 | |
|         notin:        '2209',
 | |
|         owns:         '220B',
 | |
|         gg:           '226B',
 | |
|         ll:           '226A',
 | |
|         sim:          '223C',
 | |
|         simeq:        '2243',
 | |
|         perp:         '22A5',
 | |
|         equiv:        '2261',
 | |
|         asymp:        '224D',
 | |
|         smile:        '2323',
 | |
|         frown:        '2322',
 | |
|         ne:           '2260',
 | |
|         neq:          '2260',
 | |
|         cong:         '2245',
 | |
|         doteq:        '2250',
 | |
|         bowtie:       '22C8',
 | |
|         models:       '22A8',
 | |
|         
 | |
|         notChar:      '29F8',
 | |
|         
 | |
|         
 | |
|         // Arrows
 | |
|         Leftrightarrow:     '21D4',
 | |
|         Leftarrow:          '21D0',
 | |
|         Rightarrow:         '21D2',
 | |
|         leftrightarrow:     '2194',
 | |
|         leftarrow:          '2190',
 | |
|         gets:               '2190',
 | |
|         rightarrow:         '2192',
 | |
|         to:                 '2192',
 | |
|         mapsto:             '21A6',
 | |
|         leftharpoonup:      '21BC',
 | |
|         leftharpoondown:    '21BD',
 | |
|         rightharpoonup:     '21C0',
 | |
|         rightharpoondown:   '21C1',
 | |
|         nearrow:            '2197',
 | |
|         searrow:            '2198',
 | |
|         nwarrow:            '2196',
 | |
|         swarrow:            '2199',
 | |
|         rightleftharpoons:  '21CC',
 | |
|         hookrightarrow:     '21AA',
 | |
|         hookleftarrow:      '21A9',
 | |
|         longleftarrow:      '27F5',
 | |
|         Longleftarrow:      '27F8',
 | |
|         longrightarrow:     '27F6',
 | |
|         Longrightarrow:     '27F9',
 | |
|         Longleftrightarrow: '27FA',
 | |
|         longleftrightarrow: '27F7',
 | |
|         longmapsto:         '27FC',
 | |
|         
 | |
|         
 | |
|         // Misc.
 | |
|         ldots:            '2026',
 | |
|         cdots:            '22EF',
 | |
|         vdots:            '22EE',
 | |
|         ddots:            '22F1',
 | |
|         dotsc:            '2026',  // dots with commas
 | |
|         dotsb:            '22EF',  // dots with binary ops and relations
 | |
|         dotsm:            '22EF',  // dots with multiplication
 | |
|         dotsi:            '22EF',  // dots with integrals
 | |
|         dotso:            '2026',  // other dots
 | |
|         
 | |
|         ldotp:            ['002E', {texClass: MML.TEXCLASS.PUNCT}],
 | |
|         cdotp:            ['22C5', {texClass: MML.TEXCLASS.PUNCT}],
 | |
|         colon:            ['003A', {texClass: MML.TEXCLASS.PUNCT}]
 | |
|       },
 | |
|       
 | |
|       mathchar7: {
 | |
|         Gamma:        '0393',
 | |
|         Delta:        '0394',
 | |
|         Theta:        '0398',
 | |
|         Lambda:       '039B',
 | |
|         Xi:           '039E',
 | |
|         Pi:           '03A0',
 | |
|         Sigma:        '03A3',
 | |
|         Upsilon:      '03A5',
 | |
|         Phi:          '03A6',
 | |
|         Psi:          '03A8',
 | |
|         Omega:        '03A9',
 | |
|         
 | |
|         '_':          '005F',
 | |
|         '#':          '0023',
 | |
|         '$':          '0024',
 | |
|         '%':          '0025',
 | |
|         '&':          '0026',
 | |
|         And:          '0026'
 | |
|       },
 | |
|       
 | |
|       delimiter: {
 | |
|         '(':                '(',
 | |
|         ')':                ')',
 | |
|         '[':                '[',
 | |
|         ']':                ']',
 | |
|         '<':                '27E8',
 | |
|         '>':                '27E9',
 | |
|         '\\lt':             '27E8',
 | |
|         '\\gt':             '27E9',
 | |
|         '/':                '/',
 | |
|         '|':                ['|',{texClass:MML.TEXCLASS.ORD}],
 | |
|         '.':                '',
 | |
|         '\\\\':             '\\',
 | |
|         '\\lmoustache':     '23B0',  // non-standard
 | |
|         '\\rmoustache':     '23B1',  // non-standard
 | |
|         '\\lgroup':         '27EE',  // non-standard
 | |
|         '\\rgroup':         '27EF',  // non-standard
 | |
|         '\\arrowvert':      '23D0',
 | |
|         '\\Arrowvert':      '2016',
 | |
|         '\\bracevert':      '23AA',  // non-standard
 | |
|         '\\Vert':           ['2225',{texClass:MML.TEXCLASS.ORD}],
 | |
|         '\\|':              ['2225',{texClass:MML.TEXCLASS.ORD}],
 | |
|         '\\vert':           ['|',{texClass:MML.TEXCLASS.ORD}],
 | |
|         '\\uparrow':        '2191',
 | |
|         '\\downarrow':      '2193',
 | |
|         '\\updownarrow':    '2195',
 | |
|         '\\Uparrow':        '21D1',
 | |
|         '\\Downarrow':      '21D3',
 | |
|         '\\Updownarrow':    '21D5',
 | |
|         '\\backslash':      '\\',
 | |
|         '\\rangle':         '27E9',
 | |
|         '\\langle':         '27E8',
 | |
|         '\\rbrace':         '}',
 | |
|         '\\lbrace':         '{',
 | |
|         '\\}':              '}',
 | |
|         '\\{':              '{',
 | |
|         '\\rceil':          '2309',
 | |
|         '\\lceil':          '2308',
 | |
|         '\\rfloor':         '230B',
 | |
|         '\\lfloor':         '230A',
 | |
|         '\\lbrack':         '[',
 | |
|         '\\rbrack':         ']'
 | |
|       },
 | |
|       
 | |
|       macros: {
 | |
|         displaystyle:      ['SetStyle','D',true,0],
 | |
|         textstyle:         ['SetStyle','T',false,0],
 | |
|         scriptstyle:       ['SetStyle','S',false,1],
 | |
|         scriptscriptstyle: ['SetStyle','SS',false,2],
 | |
|         
 | |
|         rm:                ['SetFont',MML.VARIANT.NORMAL],
 | |
|         mit:               ['SetFont',MML.VARIANT.ITALIC],
 | |
|         oldstyle:          ['SetFont',MML.VARIANT.OLDSTYLE],
 | |
|         cal:               ['SetFont',MML.VARIANT.CALIGRAPHIC],
 | |
|         it:                ['SetFont',"-tex-mathit"], // needs special handling
 | |
|         bf:                ['SetFont',MML.VARIANT.BOLD],
 | |
|         bbFont:            ['SetFont',MML.VARIANT.DOUBLESTRUCK],
 | |
|         scr:               ['SetFont',MML.VARIANT.SCRIPT],
 | |
|         frak:              ['SetFont',MML.VARIANT.FRAKTUR],
 | |
|         sf:                ['SetFont',MML.VARIANT.SANSSERIF],
 | |
|         tt:                ['SetFont',MML.VARIANT.MONOSPACE],
 | |
| 
 | |
| //      font:
 | |
|         
 | |
|         tiny:              ['SetSize',0.5],
 | |
|         Tiny:              ['SetSize',0.6],  // non-standard
 | |
|         scriptsize:        ['SetSize',0.7],
 | |
|         small:             ['SetSize',0.85],
 | |
|         normalsize:        ['SetSize',1.0],
 | |
|         large:             ['SetSize',1.2],
 | |
|         Large:             ['SetSize',1.44],
 | |
|         LARGE:             ['SetSize',1.73],
 | |
|         huge:              ['SetSize',2.07],
 | |
|         Huge:              ['SetSize',2.49],
 | |
|         
 | |
|         arcsin:            ['NamedFn'],
 | |
|         arccos:            ['NamedFn'],
 | |
|         arctan:            ['NamedFn'],
 | |
|         arg:               ['NamedFn'],
 | |
|         cos:               ['NamedFn'],
 | |
|         cosh:              ['NamedFn'],
 | |
|         cot:               ['NamedFn'],
 | |
|         coth:              ['NamedFn'],
 | |
|         csc:               ['NamedFn'],
 | |
|         deg:               ['NamedFn'],
 | |
|         det:                'NamedOp',
 | |
|         dim:               ['NamedFn'],
 | |
|         exp:               ['NamedFn'],
 | |
|         gcd:                'NamedOp',
 | |
|         hom:               ['NamedFn'],
 | |
|         inf:                'NamedOp',
 | |
|         ker:               ['NamedFn'],
 | |
|         lg:                ['NamedFn'],
 | |
|         lim:                'NamedOp',
 | |
|         liminf:            ['NamedOp','lim inf'],
 | |
|         limsup:            ['NamedOp','lim sup'],
 | |
|         ln:                ['NamedFn'],
 | |
|         log:               ['NamedFn'],
 | |
|         max:                'NamedOp',
 | |
|         min:                'NamedOp',
 | |
|         Pr:                 'NamedOp',
 | |
|         sec:               ['NamedFn'],
 | |
|         sin:               ['NamedFn'],
 | |
|         sinh:              ['NamedFn'],
 | |
|         sup:                'NamedOp',
 | |
|         tan:               ['NamedFn'],
 | |
|         tanh:              ['NamedFn'],
 | |
|         
 | |
|         limits:            ['Limits',1],
 | |
|         nolimits:          ['Limits',0],
 | |
| 
 | |
|         overline:            ['UnderOver','00AF',null,1],
 | |
|         underline:           ['UnderOver','005F'],
 | |
|         overbrace:           ['UnderOver','23DE',1],
 | |
|         underbrace:          ['UnderOver','23DF',1],
 | |
|         overparen:           ['UnderOver','23DC'],
 | |
|         underparen:          ['UnderOver','23DD'],
 | |
|         overrightarrow:      ['UnderOver','2192'],
 | |
|         underrightarrow:     ['UnderOver','2192'],
 | |
|         overleftarrow:       ['UnderOver','2190'],
 | |
|         underleftarrow:      ['UnderOver','2190'],
 | |
|         overleftrightarrow:  ['UnderOver','2194'],
 | |
|         underleftrightarrow: ['UnderOver','2194'],
 | |
| 
 | |
|         overset:            'Overset',
 | |
|         underset:           'Underset',
 | |
|         stackrel:           ['Macro','\\mathrel{\\mathop{#2}\\limits^{#1}}',2],
 | |
|           
 | |
|         over:               'Over',
 | |
|         overwithdelims:     'Over',
 | |
|         atop:               'Over',
 | |
|         atopwithdelims:     'Over',
 | |
|         above:              'Over',
 | |
|         abovewithdelims:    'Over',
 | |
|         brace:             ['Over','{','}'],
 | |
|         brack:             ['Over','[',']'],
 | |
|         choose:            ['Over','(',')'],
 | |
|         
 | |
|         frac:               'Frac',
 | |
|         sqrt:               'Sqrt',
 | |
|         root:               'Root',
 | |
|         uproot:            ['MoveRoot','upRoot'],
 | |
|         leftroot:          ['MoveRoot','leftRoot'],
 | |
|         
 | |
|         left:               'LeftRight',
 | |
|         right:              'LeftRight',
 | |
|         middle:             'Middle',
 | |
| 
 | |
|         llap:               'Lap',
 | |
|         rlap:               'Lap',
 | |
|         raise:              'RaiseLower',
 | |
|         lower:              'RaiseLower',
 | |
|         moveleft:           'MoveLeftRight',
 | |
|         moveright:          'MoveLeftRight',
 | |
| 
 | |
|         ',':               ['Spacer',MML.LENGTH.THINMATHSPACE],
 | |
|         ':':               ['Spacer',MML.LENGTH.MEDIUMMATHSPACE],  // for LaTeX
 | |
|         '>':               ['Spacer',MML.LENGTH.MEDIUMMATHSPACE],
 | |
|         ';':               ['Spacer',MML.LENGTH.THICKMATHSPACE],
 | |
|         '!':               ['Spacer',MML.LENGTH.NEGATIVETHINMATHSPACE],
 | |
|         enspace:           ['Spacer',".5em"],
 | |
|         quad:              ['Spacer',"1em"],
 | |
|         qquad:             ['Spacer',"2em"],
 | |
|         thinspace:         ['Spacer',MML.LENGTH.THINMATHSPACE],
 | |
|         negthinspace:      ['Spacer',MML.LENGTH.NEGATIVETHINMATHSPACE],
 | |
|     
 | |
|         hskip:              'Hskip',
 | |
|         hspace:             'Hskip',
 | |
|         kern:               'Hskip',
 | |
|         mskip:              'Hskip',
 | |
|         mspace:             'Hskip',
 | |
|         mkern:              'Hskip',
 | |
|         Rule:              ['Rule'],
 | |
|         Space:             ['Rule','blank'],
 | |
|     
 | |
|         big:               ['MakeBig',MML.TEXCLASS.ORD,0.85],
 | |
|         Big:               ['MakeBig',MML.TEXCLASS.ORD,1.15],
 | |
|         bigg:              ['MakeBig',MML.TEXCLASS.ORD,1.45],
 | |
|         Bigg:              ['MakeBig',MML.TEXCLASS.ORD,1.75],
 | |
|         bigl:              ['MakeBig',MML.TEXCLASS.OPEN,0.85],
 | |
|         Bigl:              ['MakeBig',MML.TEXCLASS.OPEN,1.15],
 | |
|         biggl:             ['MakeBig',MML.TEXCLASS.OPEN,1.45],
 | |
|         Biggl:             ['MakeBig',MML.TEXCLASS.OPEN,1.75],
 | |
|         bigr:              ['MakeBig',MML.TEXCLASS.CLOSE,0.85],
 | |
|         Bigr:              ['MakeBig',MML.TEXCLASS.CLOSE,1.15],
 | |
|         biggr:             ['MakeBig',MML.TEXCLASS.CLOSE,1.45],
 | |
|         Biggr:             ['MakeBig',MML.TEXCLASS.CLOSE,1.75],
 | |
|         bigm:              ['MakeBig',MML.TEXCLASS.REL,0.85],
 | |
|         Bigm:              ['MakeBig',MML.TEXCLASS.REL,1.15],
 | |
|         biggm:             ['MakeBig',MML.TEXCLASS.REL,1.45],
 | |
|         Biggm:             ['MakeBig',MML.TEXCLASS.REL,1.75],
 | |
| 
 | |
|         mathord:           ['TeXAtom',MML.TEXCLASS.ORD],
 | |
|         mathop:            ['TeXAtom',MML.TEXCLASS.OP],
 | |
|         mathopen:          ['TeXAtom',MML.TEXCLASS.OPEN],
 | |
|         mathclose:         ['TeXAtom',MML.TEXCLASS.CLOSE],
 | |
|         mathbin:           ['TeXAtom',MML.TEXCLASS.BIN],
 | |
|         mathrel:           ['TeXAtom',MML.TEXCLASS.REL],
 | |
|         mathpunct:         ['TeXAtom',MML.TEXCLASS.PUNCT],
 | |
|         mathinner:         ['TeXAtom',MML.TEXCLASS.INNER],
 | |
| 
 | |
|         vcenter:           ['TeXAtom',MML.TEXCLASS.VCENTER],
 | |
| 
 | |
|         mathchoice:        ['Extension','mathchoice'],
 | |
|         buildrel:           'BuildRel',
 | |
|     
 | |
|         hbox:               ['HBox',0],
 | |
|         text:               'HBox',
 | |
|         mbox:               ['HBox',0],
 | |
|         fbox:               'FBox',
 | |
| 
 | |
|         strut:              'Strut',
 | |
|         mathstrut:         ['Macro','\\vphantom{(}'],
 | |
|         phantom:            'Phantom',
 | |
|         vphantom:          ['Phantom',1,0],
 | |
|         hphantom:          ['Phantom',0,1],
 | |
|         smash:              'Smash',
 | |
|     
 | |
|         acute:             ['Accent', "00B4"],  // or 0301 or 02CA
 | |
|         grave:             ['Accent', "0060"],  // or 0300 or 02CB
 | |
|         ddot:              ['Accent', "00A8"],  // or 0308
 | |
|         tilde:             ['Accent', "007E"],  // or 0303 or 02DC
 | |
|         bar:               ['Accent', "00AF"],  // or 0304 or 02C9
 | |
|         breve:             ['Accent', "02D8"],  // or 0306
 | |
|         check:             ['Accent', "02C7"],  // or 030C
 | |
|         hat:               ['Accent', "005E"],  // or 0302 or 02C6
 | |
|         vec:               ['Accent', "2192"],  // or 20D7
 | |
|         dot:               ['Accent', "02D9"],  // or 0307
 | |
|         widetilde:         ['Accent', "007E",1], // or 0303 or 02DC
 | |
|         widehat:           ['Accent', "005E",1], // or 0302 or 02C6
 | |
| 
 | |
|         matrix:             'Matrix',
 | |
|         array:              'Matrix',
 | |
|         pmatrix:           ['Matrix','(',')'],
 | |
|         cases:             ['Matrix','{','',"left left",null,".1em",null,true],
 | |
|         eqalign:           ['Matrix',null,null,"right left",MML.LENGTH.THICKMATHSPACE,".5em",'D'],
 | |
|         displaylines:      ['Matrix',null,null,"center",null,".5em",'D'],
 | |
|         cr:                 'Cr',
 | |
|         '\\':               'CrLaTeX',
 | |
|         newline:            'Cr',
 | |
|         hline:             ['HLine','solid'],
 | |
|         hdashline:         ['HLine','dashed'],
 | |
| //      noalign:            'HandleNoAlign',
 | |
|         eqalignno:         ['Matrix',null,null,"right left",MML.LENGTH.THICKMATHSPACE,".5em",'D',null,"right"],
 | |
|         leqalignno:        ['Matrix',null,null,"right left",MML.LENGTH.THICKMATHSPACE,".5em",'D',null,"left"],
 | |
|         hfill:              'HFill',
 | |
|         hfil:               'HFill',   // \hfil treated as \hfill for now
 | |
|         hfilll:             'HFill',   // \hfilll treated as \hfill for now
 | |
| 
 | |
|         //  TeX substitution macros
 | |
|         bmod:              ['Macro','\\mmlToken{mo}[lspace="thickmathspace" rspace="thickmathspace"]{mod}'],
 | |
|         pmod:              ['Macro','\\pod{\\mmlToken{mi}{mod}\\kern 6mu #1}',1],
 | |
|         mod:               ['Macro','\\mathchoice{\\kern18mu}{\\kern12mu}{\\kern12mu}{\\kern12mu}\\mmlToken{mi}{mod}\\,\\,#1',1],
 | |
|         pod:               ['Macro','\\mathchoice{\\kern18mu}{\\kern8mu}{\\kern8mu}{\\kern8mu}(#1)',1],
 | |
|         iff:               ['Macro','\\;\\Longleftrightarrow\\;'],
 | |
|         skew:              ['Macro','{{#2{#3\\mkern#1mu}\\mkern-#1mu}{}}',3],
 | |
|         mathcal:           ['Macro','{\\cal #1}',1],
 | |
|         mathscr:           ['Macro','{\\scr #1}',1],
 | |
|         mathrm:            ['Macro','{\\rm #1}',1],
 | |
|         mathbf:            ['Macro','{\\bf #1}',1],
 | |
|         mathbb:            ['Macro','{\\bbFont #1}',1],
 | |
|         Bbb:               ['Macro','{\\bbFont #1}',1],
 | |
|         mathit:            ['Macro','{\\it #1}',1],
 | |
|         mathfrak:          ['Macro','{\\frak #1}',1],
 | |
|         mathsf:            ['Macro','{\\sf #1}',1],
 | |
|         mathtt:            ['Macro','{\\tt #1}',1],
 | |
|         textrm:            ['Macro','\\mathord{\\rm\\text{#1}}',1],
 | |
|         textit:            ['Macro','\\mathord{\\it\\text{#1}}',1],
 | |
|         textbf:            ['Macro','\\mathord{\\bf\\text{#1}}',1],
 | |
|         textsf:            ['Macro','\\mathord{\\sf\\text{#1}}',1],
 | |
|         texttt:            ['Macro','\\mathord{\\tt\\text{#1}}',1],
 | |
|         pmb:               ['Macro','\\rlap{#1}\\kern1px{#1}',1],
 | |
|         TeX:               ['Macro','T\\kern-.14em\\lower.5ex{E}\\kern-.115em X'],
 | |
|         LaTeX:             ['Macro','L\\kern-.325em\\raise.21em{\\scriptstyle{A}}\\kern-.17em\\TeX'],
 | |
|         ' ':               ['Macro','\\text{ }'],
 | |
| 
 | |
|         //  Specially handled
 | |
|         not:                'Not',
 | |
|         dots:               'Dots',
 | |
|         space:              'Tilde',
 | |
|         '\u00A0':           'Tilde',
 | |
|         
 | |
| 
 | |
|         //  LaTeX
 | |
|         begin:              'BeginEnd',
 | |
|         end:                'BeginEnd',
 | |
| 
 | |
|         newcommand:        ['Extension','newcommand'],
 | |
|         renewcommand:      ['Extension','newcommand'],
 | |
|         newenvironment:    ['Extension','newcommand'],
 | |
|         renewenvironment:  ['Extension','newcommand'],
 | |
|         def:               ['Extension','newcommand'],
 | |
|         let:               ['Extension','newcommand'],
 | |
|         
 | |
|         verb:              ['Extension','verb'],
 | |
|         
 | |
|         boldsymbol:        ['Extension','boldsymbol'],
 | |
|         
 | |
|         tag:               ['Extension','AMSmath'],
 | |
|         notag:             ['Extension','AMSmath'],
 | |
|         label:             ['Extension','AMSmath'],
 | |
|         ref:               ['Extension','AMSmath'],
 | |
|         eqref:             ['Extension','AMSmath'],
 | |
|         nonumber:          ['Macro','\\notag'],
 | |
| 
 | |
|         //  Extensions to TeX
 | |
|         unicode:           ['Extension','unicode'],
 | |
|         color:              'Color',
 | |
|         
 | |
|         href:              ['Extension','HTML'],
 | |
|         'class':           ['Extension','HTML'],
 | |
|         style:             ['Extension','HTML'],
 | |
|         cssId:             ['Extension','HTML'],
 | |
|         bbox:              ['Extension','bbox'],
 | |
|     
 | |
|         mmlToken:           'MmlToken',
 | |
| 
 | |
|         require:            'Require'
 | |
| 
 | |
|       },
 | |
|       
 | |
|       environment: {
 | |
|         array:        ['AlignedArray'],
 | |
|         matrix:       ['Array',null,null,null,'c'],
 | |
|         pmatrix:      ['Array',null,'(',')','c'],
 | |
|         bmatrix:      ['Array',null,'[',']','c'],
 | |
|         Bmatrix:      ['Array',null,'\\{','\\}','c'],
 | |
|         vmatrix:      ['Array',null,'\\vert','\\vert','c'],
 | |
|         Vmatrix:      ['Array',null,'\\Vert','\\Vert','c'],
 | |
|         cases:        ['Array',null,'\\{','.','ll',null,".2em",'T'],
 | |
| 
 | |
|         equation:     [null,'Equation'],
 | |
|         'equation*':  [null,'Equation'],
 | |
| 
 | |
|         eqnarray:     ['ExtensionEnv',null,'AMSmath'],
 | |
|         'eqnarray*':  ['ExtensionEnv',null,'AMSmath'],
 | |
| 
 | |
|         align:        ['ExtensionEnv',null,'AMSmath'],
 | |
|         'align*':     ['ExtensionEnv',null,'AMSmath'],
 | |
|         aligned:      ['ExtensionEnv',null,'AMSmath'],
 | |
|         multline:     ['ExtensionEnv',null,'AMSmath'],
 | |
|         'multline*':  ['ExtensionEnv',null,'AMSmath'],
 | |
|         split:        ['ExtensionEnv',null,'AMSmath'],
 | |
|         gather:       ['ExtensionEnv',null,'AMSmath'],
 | |
|         'gather*':    ['ExtensionEnv',null,'AMSmath'],
 | |
|         gathered:     ['ExtensionEnv',null,'AMSmath'],
 | |
|         alignat:      ['ExtensionEnv',null,'AMSmath'],
 | |
|         'alignat*':   ['ExtensionEnv',null,'AMSmath'],
 | |
|         alignedat:    ['ExtensionEnv',null,'AMSmath']
 | |
|       },
 | |
|       
 | |
|       p_height: 1.2 / .85   // cmex10 height plus depth over .85
 | |
| 
 | |
|     });
 | |
|     
 | |
|     //
 | |
|     //  Add macros defined in the configuration
 | |
|     //
 | |
|     if (this.config.Macros) {
 | |
|       var MACROS = this.config.Macros;
 | |
|       for (var id in MACROS) {if (MACROS.hasOwnProperty(id)) {
 | |
|         if (typeof(MACROS[id]) === "string") {TEXDEF.macros[id] = ['Macro',MACROS[id]]}
 | |
|         else {TEXDEF.macros[id] = ["Macro"].concat(MACROS[id])}
 | |
|         TEXDEF.macros[id].isUser = true;
 | |
|       }}
 | |
|     }
 | |
|   };
 | |
|   
 | |
|   /************************************************************************/
 | |
|   /*
 | |
|    *   The TeX Parser
 | |
|    */
 | |
| 
 | |
|   var PARSE = MathJax.Object.Subclass({
 | |
|     Init: function (string,env) {
 | |
|       this.string = string; this.i = 0; this.macroCount = 0;
 | |
|       var ENV; if (env) {ENV = {}; for (var id in env) {if (env.hasOwnProperty(id)) {ENV[id] = env[id]}}}
 | |
|       this.stack = TEX.Stack(ENV,!!env);
 | |
|       this.Parse(); this.Push(STACKITEM.stop());
 | |
|     },
 | |
|     Parse: function () {
 | |
|       var c, n;
 | |
|       while (this.i < this.string.length) {
 | |
|         c = this.string.charAt(this.i++); n = c.charCodeAt(0);
 | |
|         if (n >= 0xD800 && n < 0xDC00) {c += this.string.charAt(this.i++)}
 | |
|         if (TEXDEF.special[c]) {this[TEXDEF.special[c]](c)}
 | |
|         else if (TEXDEF.letter.test(c)) {this.Variable(c)}
 | |
|         else if (TEXDEF.digit.test(c)) {this.Number(c)}
 | |
|         else {this.Other(c)}
 | |
|       }
 | |
|     },
 | |
|     Push: function () {this.stack.Push.apply(this.stack,arguments)},
 | |
|     mml: function () {
 | |
|       if (this.stack.Top().type !== "mml") {return null}
 | |
|       return this.stack.Top().data[0];
 | |
|     },
 | |
|     mmlToken: function (token) {return token}, // used by boldsymbol extension
 | |
| 
 | |
|     /************************************************************************/
 | |
|     /*
 | |
|      *   Handle various token classes
 | |
|      */
 | |
| 
 | |
|     /*
 | |
|      *  Lookup a control-sequence and process it
 | |
|      */
 | |
|     ControlSequence: function (c) {
 | |
|       var name = this.GetCS(), macro = this.csFindMacro(name);
 | |
|       if (macro) {
 | |
|         if (!(macro instanceof Array)) {macro = [macro]}
 | |
|         var fn = macro[0]; if (!(fn instanceof Function)) {fn = this[fn]}
 | |
|         fn.apply(this,[c+name].concat(macro.slice(1)));
 | |
|       } else if (TEXDEF.mathchar0mi[name])            {this.csMathchar0mi(name,TEXDEF.mathchar0mi[name])}
 | |
|         else if (TEXDEF.mathchar0mo[name])            {this.csMathchar0mo(name,TEXDEF.mathchar0mo[name])}
 | |
|         else if (TEXDEF.mathchar7[name])              {this.csMathchar7(name,TEXDEF.mathchar7[name])}
 | |
|         else if (TEXDEF.delimiter["\\"+name] != null) {this.csDelimiter(name,TEXDEF.delimiter["\\"+name])}
 | |
|         else                                          {this.csUndefined(c+name)}
 | |
|     },
 | |
|     //
 | |
|     //  Look up a macro in the macros list
 | |
|     //  (overridden in begingroup extension)
 | |
|     //
 | |
|     csFindMacro: function (name) {return TEXDEF.macros[name]},
 | |
|     //
 | |
|     //  Handle normal mathchar (as an mi)
 | |
|     //
 | |
|     csMathchar0mi: function (name,mchar) {
 | |
|       var def = {mathvariant: MML.VARIANT.ITALIC};
 | |
|       if (mchar instanceof Array) {def = mchar[1]; mchar = mchar[0]}
 | |
|       this.Push(this.mmlToken(MML.mi(MML.entity("#x"+mchar)).With(def)));
 | |
|     },
 | |
|     //
 | |
|     //  Handle normal mathchar (as an mo)
 | |
|     //
 | |
|     csMathchar0mo: function (name,mchar) {
 | |
|       var def = {stretchy: false};
 | |
|       if (mchar instanceof Array) {def = mchar[1]; def.stretchy = false; mchar = mchar[0]}
 | |
|       this.Push(this.mmlToken(MML.mo(MML.entity("#x"+mchar)).With(def)));
 | |
|     },
 | |
|     //
 | |
|     //  Handle mathchar in current family
 | |
|     //
 | |
|     csMathchar7: function (name,mchar) {
 | |
|       var def = {mathvariant: MML.VARIANT.NORMAL};
 | |
|       if (mchar instanceof Array) {def = mchar[1]; mchar = mchar[0]}
 | |
|       if (this.stack.env.font) {def.mathvariant = this.stack.env.font}
 | |
|       this.Push(this.mmlToken(MML.mi(MML.entity("#x"+mchar)).With(def)));
 | |
|     },
 | |
|     //
 | |
|     //  Handle delimiter
 | |
|     //
 | |
|     csDelimiter: function (name,delim) {
 | |
|       var def = {};
 | |
|       if (delim instanceof Array) {def = delim[1]; delim = delim[0]}
 | |
|       if (delim.length === 4) {delim = MML.entity('#x'+delim)} else {delim = MML.chars(delim)}
 | |
|       this.Push(this.mmlToken(MML.mo(delim).With({fence: false, stretchy: false}).With(def)));
 | |
|     },
 | |
|     //
 | |
|     //  Handle undefined control sequence
 | |
|     //  (overridden in noUndefined extension)
 | |
|     //
 | |
|     csUndefined: function (name) {
 | |
|       TEX.Error(["UndefinedControlSequence","Undefined control sequence %1",name]);
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Handle a variable (a single letter)
 | |
|      */
 | |
|     Variable: function (c) {
 | |
|       var def = {}; if (this.stack.env.font) {def.mathvariant = this.stack.env.font}
 | |
|       this.Push(this.mmlToken(MML.mi(MML.chars(c)).With(def)));
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Determine the extent of a number (pattern may need work)
 | |
|      */
 | |
|     Number: function (c) {
 | |
|       var mml, n = this.string.slice(this.i-1).match(TEXDEF.number);
 | |
|       if (n) {mml = MML.mn(n[0].replace(/[{}]/g,"")); this.i += n[0].length - 1}
 | |
|         else {mml = MML.mo(MML.chars(c))}
 | |
|       if (this.stack.env.font) {mml.mathvariant = this.stack.env.font}
 | |
|       this.Push(this.mmlToken(mml));
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Handle { and }
 | |
|      */
 | |
|     Open: function (c) {this.Push(STACKITEM.open())},
 | |
|     Close: function (c) {this.Push(STACKITEM.close())},
 | |
|     
 | |
|     /*
 | |
|      *  Handle tilde and spaces
 | |
|      */
 | |
|     Tilde: function (c) {this.Push(MML.mtext(MML.chars(NBSP)))},
 | |
|     Space: function (c) {},
 | |
|     
 | |
|     /*
 | |
|      *  Handle ^, _, and '
 | |
|      */
 | |
|     Superscript: function (c) {
 | |
|       if (this.GetNext().match(/\d/)) // don't treat numbers as a unit
 | |
|         {this.string = this.string.substr(0,this.i+1)+" "+this.string.substr(this.i+1)}
 | |
|       var primes, base, top = this.stack.Top();
 | |
|       if (top.type === "prime") {base = top.data[0]; primes = top.data[1]; this.stack.Pop()}
 | |
|         else {base = this.stack.Prev(); if (!base) {base = MML.mi("")}}
 | |
|       if (base.isEmbellishedWrapper) {base = base.data[0].data[0]}
 | |
|       var movesupsub = base.movesupsub, position = base.sup;
 | |
|       if ((base.type === "msubsup" && base.data[base.sup]) ||
 | |
|           (base.type === "munderover" && base.data[base.over] && !base.subsupOK))
 | |
|            {TEX.Error(["DoubleExponent","Double exponent: use braces to clarify"])}
 | |
|       if (base.type !== "msubsup") {
 | |
|         if (movesupsub) {
 | |
|           if (base.type !== "munderover" || base.data[base.over]) {
 | |
|             if (base.movablelimits && base.isa(MML.mi)) {base = this.mi2mo(base)}
 | |
|             base = MML.munderover(base,null,null).With({movesupsub:true})
 | |
|           }
 | |
|           position = base.over;
 | |
|         } else {
 | |
|           base = MML.msubsup(base,null,null);
 | |
|           position = base.sup;
 | |
|         }
 | |
|       }
 | |
|       this.Push(STACKITEM.subsup(base).With({
 | |
|         position: position, primes: primes, movesupsub: movesupsub
 | |
|       }));
 | |
|     },
 | |
|     Subscript: function (c) {
 | |
|       if (this.GetNext().match(/\d/)) // don't treat numbers as a unit
 | |
|         {this.string = this.string.substr(0,this.i+1)+" "+this.string.substr(this.i+1)}
 | |
|       var primes, base, top = this.stack.Top();
 | |
|       if (top.type === "prime") {base = top.data[0]; primes = top.data[1]; this.stack.Pop()}
 | |
|         else {base = this.stack.Prev(); if (!base) {base = MML.mi("")}}
 | |
|       if (base.isEmbellishedWrapper) {base = base.data[0].data[0]}
 | |
|       var movesupsub = base.movesupsub, position = base.sub;
 | |
|       if ((base.type === "msubsup" && base.data[base.sub]) ||
 | |
|           (base.type === "munderover" && base.data[base.under] && !base.subsupOK))
 | |
|            {TEX.Error(["DoubleSubscripts","Double subscripts: use braces to clarify"])}
 | |
|       if (base.type !== "msubsup") {
 | |
|         if (movesupsub) {
 | |
|           if (base.type !== "munderover" || base.data[base.under]) {
 | |
|             if (base.movablelimits && base.isa(MML.mi)) {base = this.mi2mo(base)}
 | |
|             base = MML.munderover(base,null,null).With({movesupsub:true})
 | |
|           }
 | |
|           position = base.under;
 | |
|         } else {
 | |
|           base = MML.msubsup(base,null,null);
 | |
|           position = base.sub;
 | |
|         }
 | |
|       }
 | |
|       this.Push(STACKITEM.subsup(base).With({
 | |
|         position: position, primes: primes, movesupsub: movesupsub
 | |
|       }));
 | |
|     },
 | |
|     PRIME: "\u2032", SMARTQUOTE: "\u2019",
 | |
|     Prime: function (c) {
 | |
|       var base = this.stack.Prev(); if (!base) {base = MML.mi()}
 | |
|       if (base.type === "msubsup" && base.data[base.sup]) {
 | |
|         TEX.Error(["DoubleExponentPrime",
 | |
|                    "Prime causes double exponent: use braces to clarify"]);
 | |
|       }
 | |
|       var sup = ""; this.i--;
 | |
|       do {sup += this.PRIME; this.i++, c = this.GetNext()}
 | |
|         while (c === "'" || c === this.SMARTQUOTE);
 | |
|       sup = ["","\u2032","\u2033","\u2034","\u2057"][sup.length] || sup;
 | |
|       this.Push(STACKITEM.prime(base,this.mmlToken(MML.mo(sup))));
 | |
|     },
 | |
|     mi2mo: function (mi) {
 | |
|       var mo = MML.mo();  mo.Append.apply(mo,mi.data); var id;
 | |
|       for (id in mo.defaults)
 | |
|         {if (mo.defaults.hasOwnProperty(id) && mi[id] != null) {mo[id] = mi[id]}}
 | |
|       for (id in MML.copyAttributes)
 | |
|         {if (MML.copyAttributes.hasOwnProperty(id) && mi[id] != null) {mo[id] = mi[id]}}
 | |
|       mo.lspace = mo.rspace = "0";  // prevent mo from having space in NativeMML
 | |
|       mo.useMMLspacing &= ~(mo.SPACE_ATTR.lspace | mo.SPACE_ATTR.rspace);  // don't count these explicit settings
 | |
|       return mo;
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Handle comments
 | |
|      */
 | |
|     Comment: function (c) {
 | |
|       while (this.i < this.string.length && this.string.charAt(this.i) != "\n") {this.i++}
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Handle hash marks outside of definitions
 | |
|      */
 | |
|     Hash: function (c) {
 | |
|       TEX.Error(["CantUseHash1",
 | |
|                  "You can't use 'macro parameter character #' in math mode"]);
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Handle other characters (as <mo> elements)
 | |
|      */
 | |
|     Other: function (c) {
 | |
|       var def, mo;
 | |
|       if (this.stack.env.font) {def = {mathvariant: this.stack.env.font}}
 | |
|       if (TEXDEF.remap[c]) {
 | |
|         c = TEXDEF.remap[c];
 | |
|         if (c instanceof Array) {def = c[1]; c = c[0]}
 | |
|         mo = MML.mo(MML.entity('#x'+c)).With(def);
 | |
|       } else {
 | |
|         mo = MML.mo(c).With(def);
 | |
|       }
 | |
|       if (mo.autoDefault("stretchy",true)) {mo.stretchy = false}
 | |
|       if (mo.autoDefault("texClass",true) == "") {mo = MML.TeXAtom(mo)}
 | |
|       this.Push(this.mmlToken(mo));
 | |
|     },
 | |
|     
 | |
|     /************************************************************************/
 | |
|     /*
 | |
|      *   Macros
 | |
|      */
 | |
|     
 | |
|     SetFont: function (name,font) {this.stack.env.font = font},
 | |
|     SetStyle: function (name,texStyle,style,level) {
 | |
|       this.stack.env.style = texStyle; this.stack.env.level = level;
 | |
|       this.Push(STACKITEM.style().With({styles: {displaystyle: style, scriptlevel: level}}));
 | |
|     },
 | |
|     SetSize: function (name,size) {
 | |
|       this.stack.env.size = size;
 | |
|       this.Push(STACKITEM.style().With({styles: {mathsize: size+"em"}})); // convert to absolute?
 | |
|     },
 | |
| 
 | |
|     Color: function (name) {
 | |
|       var color = this.GetArgument(name);
 | |
|       var old = this.stack.env.color; this.stack.env.color = color;
 | |
|       var math = this.ParseArg(name);
 | |
|       if (old) {this.stack.env.color} else {delete this.stack.env.color}
 | |
|       this.Push(MML.mstyle(math).With({mathcolor: color}));
 | |
|     },
 | |
|     
 | |
|     Spacer: function (name,space) {
 | |
|       this.Push(MML.mspace().With({width: space, mathsize: MML.SIZE.NORMAL, scriptlevel:0}));
 | |
|     },
 | |
|     
 | |
|     LeftRight: function (name) {
 | |
|       this.Push(STACKITEM[name.substr(1)]().With({delim: this.GetDelimiter(name)}));
 | |
|     },
 | |
|     
 | |
|     Middle: function (name) {
 | |
|       var delim = this.GetDelimiter(name);
 | |
|       if (this.stack.Top().type !== "left")
 | |
|         {TEX.Error(["MisplacedMiddle","%1 must be within \\left and \\right",name])}
 | |
|       this.Push(MML.mo(delim).With({stretchy:true}));
 | |
|     },
 | |
|     
 | |
|     NamedFn: function (name,id) {
 | |
|       if (!id) {id = name.substr(1)};
 | |
|       var mml = MML.mi(id).With({texClass: MML.TEXCLASS.OP});
 | |
|       this.Push(STACKITEM.fn(this.mmlToken(mml)));
 | |
|     },
 | |
|     NamedOp: function (name,id) {
 | |
|       if (!id) {id = name.substr(1)};
 | |
|       id = id.replace(/ /,"\u2006");
 | |
|       var mml = MML.mo(id).With({
 | |
|         movablelimits: true,
 | |
|         movesupsub: true,
 | |
|         form: MML.FORM.PREFIX,
 | |
|         texClass: MML.TEXCLASS.OP
 | |
|       });
 | |
|       mml.useMMLspacing &= ~mml.SPACE_ATTR.form;  // don't count this explicit form setting
 | |
|       this.Push(this.mmlToken(mml));
 | |
|     },
 | |
|     Limits: function (name,limits) {
 | |
|       var op = this.stack.Prev("nopop");
 | |
|       if (!op || (op.Get("texClass") !== MML.TEXCLASS.OP && op.movesupsub == null))
 | |
|         {TEX.Error(["MisplacedLimits","%1 is allowed only on operators",name])}
 | |
|       var top = this.stack.Top();
 | |
|       if (op.type === "munderover" && !limits) {
 | |
|         op = top.data[top.data.length-1] = MML.msubsup.apply(MML.subsup,op.data);
 | |
|       } else if (op.type === "msubsup" && limits) {
 | |
|         op = top.data[top.data.length-1] = MML.munderover.apply(MML.underover,op.data);
 | |
|       }
 | |
|       op.movesupsub = (limits ? true : false);
 | |
|       op.Core().movablelimits = false;
 | |
|       if (op.movablelimits) op.movablelimits = false;
 | |
|     },
 | |
|     
 | |
|     Over: function (name,open,close) {
 | |
|       var mml = STACKITEM.over().With({name: name});
 | |
|       if (open || close) {
 | |
|         mml.open = open; mml.close = close;
 | |
|       } else if (name.match(/withdelims$/)) {
 | |
|         mml.open  = this.GetDelimiter(name);
 | |
|         mml.close = this.GetDelimiter(name);
 | |
|       }
 | |
|       if (name.match(/^\\above/)) {mml.thickness = this.GetDimen(name)}
 | |
|       else if (name.match(/^\\atop/) || open || close) {mml.thickness = 0}
 | |
|       this.Push(mml);
 | |
|     },
 | |
| 
 | |
|     Frac: function (name) {
 | |
|       var num = this.ParseArg(name);
 | |
|       var den = this.ParseArg(name);
 | |
|       this.Push(MML.mfrac(num,den));
 | |
|     },
 | |
| 
 | |
|     Sqrt: function (name) {
 | |
|       var n = this.GetBrackets(name), arg = this.GetArgument(name);
 | |
|       if (arg === "\\frac") {arg += "{"+this.GetArgument(arg)+"}{"+this.GetArgument(arg)+"}"}
 | |
|       var mml = TEX.Parse(arg,this.stack.env).mml();
 | |
|       if (!n) {mml = MML.msqrt.apply(MML,mml.array())}
 | |
|          else {mml = MML.mroot(mml,this.parseRoot(n))}
 | |
|       this.Push(mml);
 | |
|     },
 | |
|     Root: function (name) {
 | |
|       var n = this.GetUpTo(name,"\\of");
 | |
|       var arg = this.ParseArg(name);
 | |
|       this.Push(MML.mroot(arg,this.parseRoot(n)));
 | |
|     },
 | |
|     parseRoot: function (n) {
 | |
|       var env = this.stack.env, inRoot = env.inRoot; env.inRoot = true;
 | |
|       var parser = TEX.Parse(n,env); n = parser.mml(); var global = parser.stack.global;
 | |
|       if (global.leftRoot || global.upRoot) {
 | |
|         n = MML.mpadded(n);
 | |
|         if (global.leftRoot) {n.width = global.leftRoot}
 | |
|         if (global.upRoot) {n.voffset = global.upRoot; n.height = global.upRoot}
 | |
|       }
 | |
|       env.inRoot = inRoot;
 | |
|       return n;
 | |
|     },
 | |
|     MoveRoot: function (name,id) {
 | |
|       if (!this.stack.env.inRoot)
 | |
|         {TEX.Error(["MisplacedMoveRoot","%1 can appear only within a root",name])}
 | |
|       if (this.stack.global[id])
 | |
|         {TEX.Error(["MultipleMoveRoot","Multiple use of %1",name])}
 | |
|       var n = this.GetArgument(name);
 | |
|       if (!n.match(/-?[0-9]+/))
 | |
|         {TEX.Error(["IntegerArg","The argument to %1 must be an integer",name])}
 | |
|       n = (n/15)+"em";
 | |
|       if (n.substr(0,1) !== "-") {n = "+"+n}
 | |
|       this.stack.global[id] = n;
 | |
|     },
 | |
|     
 | |
|     Accent: function (name,accent,stretchy) {
 | |
|       var c = this.ParseArg(name);
 | |
|       var def = {accent: true}; if (this.stack.env.font) {def.mathvariant = this.stack.env.font}
 | |
|       var mml = this.mmlToken(MML.mo(MML.entity("#x"+accent)).With(def));
 | |
|       mml.stretchy = (stretchy ? true : false);
 | |
|       this.Push(MML.TeXAtom(MML.munderover(c,null,mml).With({accent: true})));
 | |
|     },
 | |
|     
 | |
|     UnderOver: function (name,c,stack,noaccent) {
 | |
|       var pos = {o: "over", u: "under"}[name.charAt(1)];
 | |
|       var base = this.ParseArg(name);
 | |
|       if (base.Get("movablelimits")) {base.movablelimits = false}
 | |
|       if (base.isa(MML.munderover) && base.isEmbellished()) {
 | |
|         base.Core().With({lspace:0,rspace:0}); // get spacing right for NativeMML
 | |
|         base = MML.mrow(MML.mo().With({rspace:0}),base);  // add an empty <mi> so it's not embellished any more
 | |
|       }
 | |
|       var mml = MML.munderover(base,null,null);
 | |
|       mml.SetData(
 | |
|         mml[pos], 
 | |
|         this.mmlToken(MML.mo(MML.entity("#x"+c)).With({stretchy:true, accent:!noaccent}))
 | |
|       );
 | |
|       if (stack) {mml = MML.TeXAtom(mml).With({texClass:MML.TEXCLASS.OP, movesupsub:true})}
 | |
|       this.Push(mml.With({subsupOK:true}));
 | |
|     },
 | |
|     
 | |
|     Overset: function (name) {
 | |
|       var top = this.ParseArg(name), base = this.ParseArg(name);
 | |
|       if (base.movablelimits) base.movablelimits = false;
 | |
|       this.Push(MML.mover(base,top));
 | |
|     },
 | |
|     Underset: function (name) {
 | |
|       var bot = this.ParseArg(name), base = this.ParseArg(name);
 | |
|       if (base.movablelimits) base.movablelimits = false;
 | |
|       this.Push(MML.munder(base,bot));
 | |
|     },
 | |
|     
 | |
|     TeXAtom: function (name,mclass) {
 | |
|       var def = {texClass: mclass}, mml;
 | |
|       if (mclass == MML.TEXCLASS.OP) {
 | |
|         def.movesupsub = def.movablelimits = true;
 | |
|         var arg = this.GetArgument(name);
 | |
|         var match = arg.match(/^\s*\\rm\s+([a-zA-Z0-9 ]+)$/);
 | |
|         if (match) {
 | |
|           def.mathvariant = MML.VARIANT.NORMAL;
 | |
|           mml = STACKITEM.fn(this.mmlToken(MML.mi(match[1]).With(def)));
 | |
|         } else {
 | |
|           mml = STACKITEM.fn(MML.TeXAtom(TEX.Parse(arg,this.stack.env).mml()).With(def));
 | |
|         }
 | |
|       } else {mml = MML.TeXAtom(this.ParseArg(name)).With(def)}
 | |
|       this.Push(mml);
 | |
|     },
 | |
|     
 | |
|     MmlToken: function (name) {
 | |
|       var type = this.GetArgument(name),
 | |
|           attr = this.GetBrackets(name,"").replace(/^\s+/,""),
 | |
|           data = this.GetArgument(name),
 | |
|           def = {attrNames:[]}, match;
 | |
|       if (!MML[type] || !MML[type].prototype.isToken)
 | |
|         {TEX.Error(["NotMathMLToken","%1 is not a token element",type])}
 | |
|       while (attr !== "") {
 | |
|         match = attr.match(/^([a-z]+)\s*=\s*('[^']*'|"[^"]*"|[^ ,]*)\s*,?\s*/i);
 | |
|         if (!match)
 | |
|           {TEX.Error(["InvalidMathMLAttr","Invalid MathML attribute: %1",attr])}
 | |
|         if (MML[type].prototype.defaults[match[1]] == null && !this.MmlTokenAllow[match[1]]) {
 | |
|           TEX.Error(["UnknownAttrForElement",
 | |
|                      "%1 is not a recognized attribute for %2",
 | |
|                      match[1],type]);
 | |
|         }
 | |
|         var value = this.MmlFilterAttribute(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;
 | |
|           def.attrNames.push(match[1]);
 | |
|         }
 | |
|         attr = attr.substr(match[0].length);
 | |
|       }
 | |
|       this.Push(this.mmlToken(MML[type](data).With(def)));
 | |
|     },
 | |
|     MmlFilterAttribute: function (name,value) {return value},
 | |
|     MmlTokenAllow: {
 | |
|       fontfamily:1, fontsize:1, fontweight:1, fontstyle:1,
 | |
|       color:1, background:1,
 | |
|       id:1, "class":1, href:1, style:1
 | |
|     },
 | |
|     
 | |
|     Strut: function (name) {
 | |
|       this.Push(MML.mpadded(MML.mrow()).With({height: "8.6pt", depth: "3pt", width: 0}));
 | |
|     },
 | |
|     
 | |
|     Phantom: function (name,v,h) {
 | |
|       var box = MML.mphantom(this.ParseArg(name));
 | |
|       if (v || h) {
 | |
|         box = MML.mpadded(box);
 | |
|         if (h) {box.height = box.depth = 0}
 | |
|         if (v) {box.width = 0}
 | |
|       }
 | |
|       this.Push(MML.TeXAtom(box));
 | |
|     },
 | |
|     
 | |
|     Smash: function (name) {
 | |
|       var bt = this.trimSpaces(this.GetBrackets(name,""));
 | |
|       var smash = MML.mpadded(this.ParseArg(name));
 | |
|       switch (bt) {
 | |
|         case "b": smash.depth = 0; break;
 | |
|         case "t": smash.height = 0; break;
 | |
|         default: smash.height = smash.depth = 0;
 | |
|       }
 | |
|       this.Push(MML.TeXAtom(smash));
 | |
|     },
 | |
|     
 | |
|     Lap: function (name) {
 | |
|       var mml = MML.mpadded(this.ParseArg(name)).With({width: 0});
 | |
|       if (name === "\\llap") {mml.lspace = "-1width"}
 | |
|       this.Push(MML.TeXAtom(mml));
 | |
|     },
 | |
|     
 | |
|     RaiseLower: function (name) {
 | |
|       var h = this.GetDimen(name);
 | |
|       var item = STACKITEM.position().With({name: name, move: 'vertical'});
 | |
|       if (h.charAt(0) === '-') {h = h.slice(1); name = {raise: "\\lower", lower: "\\raise"}[name.substr(1)]}
 | |
|       if (name === "\\lower") {item.dh = '-'+h; item.dd = '+'+h} else {item.dh = '+'+h; item.dd = '-'+h}
 | |
|       this.Push(item);
 | |
|     },
 | |
|     
 | |
|     MoveLeftRight: function (name) {
 | |
|       var h = this.GetDimen(name);
 | |
|       var nh = (h.charAt(0) === '-' ? h.slice(1) : '-'+h);
 | |
|       if (name === "\\moveleft") {var tmp = h; h = nh; nh = tmp}
 | |
|       this.Push(STACKITEM.position().With({
 | |
|         name: name, move: 'horizontal',
 | |
|         left:  MML.mspace().With({width: h, mathsize: MML.SIZE.NORMAL}),
 | |
|         right: MML.mspace().With({width: nh, mathsize: MML.SIZE.NORMAL})
 | |
|       }));
 | |
|     },
 | |
|     
 | |
|     Hskip: function (name) {
 | |
|       this.Push(MML.mspace().With({width: this.GetDimen(name), mathsize: MML.SIZE.NORMAL}));
 | |
|     },
 | |
|     
 | |
|     Rule: function (name,style) {
 | |
|       var w = this.GetDimen(name),
 | |
|           h = this.GetDimen(name),
 | |
|           d = this.GetDimen(name);
 | |
|       var mml, def = {width:w, height:h, depth:d};
 | |
|       if (style !== 'blank') {
 | |
|         if (parseFloat(w) && parseFloat(h)+parseFloat(d))
 | |
|           {def.mathbackground = (this.stack.env.color || "black")}
 | |
|         mml = MML.mpadded(MML.mrow()).With(def);
 | |
|       } else {
 | |
|         mml = MML.mspace().With(def);
 | |
|       }
 | |
|       this.Push(mml);
 | |
|     },
 | |
|     
 | |
|     MakeBig: function (name,mclass,size) {
 | |
|       size *= TEXDEF.p_height;
 | |
|       size = String(size).replace(/(\.\d\d\d).+/,'$1')+"em";
 | |
|       var delim = this.GetDelimiter(name,true);
 | |
|       this.Push(MML.TeXAtom(MML.mo(delim).With({
 | |
|         minsize: size, maxsize: size,
 | |
|         fence: true, stretchy: true, symmetric: true
 | |
|       })).With({texClass: mclass}));
 | |
|     },
 | |
|     
 | |
|     BuildRel: function (name) {
 | |
|       var top = this.ParseUpTo(name,"\\over");
 | |
|       var bot = this.ParseArg(name);
 | |
|       this.Push(MML.TeXAtom(MML.munderover(bot,null,top)).With({texClass: MML.TEXCLASS.REL}));
 | |
|     },
 | |
|     
 | |
|     HBox: function (name,style) {
 | |
|       this.Push.apply(this,this.InternalMath(this.GetArgument(name),style));
 | |
|     },
 | |
|     
 | |
|     FBox: function (name) {
 | |
|       this.Push(MML.menclose.apply(MML,this.InternalMath(this.GetArgument(name))).With({notation:"box"}));
 | |
|     },
 | |
|     
 | |
|     Not: function (name) {
 | |
|       this.Push(STACKITEM.not());
 | |
|     },
 | |
|     
 | |
|     Dots: function (name) {
 | |
|       this.Push(STACKITEM.dots().With({
 | |
|         ldots: this.mmlToken(MML.mo(MML.entity("#x2026")).With({stretchy:false})),
 | |
|         cdots: this.mmlToken(MML.mo(MML.entity("#x22EF")).With({stretchy:false}))
 | |
|       }));
 | |
|     },
 | |
|     
 | |
|     Require: function (name) {
 | |
|       var file = this.GetArgument(name)
 | |
|         .replace(/.*\//,"")            // remove any leading path
 | |
|         .replace(/[^a-z0-9_.-]/ig,""); // remove illegal characters
 | |
|       this.Extension(null,file);
 | |
|     },
 | |
|     
 | |
|     Extension: function (name,file,array) {
 | |
|       if (name && !typeof(name) === "string") {name = name.name}
 | |
|       file = TEX.extensionDir+"/"+file;
 | |
|       if (!file.match(/\.js$/)) {file += ".js"}
 | |
|       if (!AJAX.loaded[AJAX.fileURL(file)]) {
 | |
|         if (name != null) {delete TEXDEF[array || 'macros'][name.replace(/^\\/,"")]}
 | |
|         HUB.RestartAfter(AJAX.Require(file));
 | |
|       }
 | |
|     },
 | |
|     
 | |
|     Macro: function (name,macro,argcount,def) {
 | |
|       if (argcount) {
 | |
|         var args = [];
 | |
|         if (def != null) {
 | |
|           var optional = this.GetBrackets(name);
 | |
|           args.push(optional == null ? def : optional);
 | |
|         }
 | |
|         for (var i = args.length; i < argcount; i++) {args.push(this.GetArgument(name))}
 | |
|         macro = this.SubstituteArgs(args,macro);
 | |
|       }
 | |
|       this.string = this.AddArgs(macro,this.string.slice(this.i));
 | |
|       this.i = 0;
 | |
|       if (++this.macroCount > TEX.config.MAXMACROS) {
 | |
|         TEX.Error(["MaxMacroSub1",
 | |
|                    "MathJax maximum macro substitution count exceeded; " +
 | |
|                    "is there a recursive macro call?"]);
 | |
|       }
 | |
|     },
 | |
|     
 | |
|     Matrix: function (name,open,close,align,spacing,vspacing,style,cases,numbered) {
 | |
|       var c = this.GetNext();
 | |
|       if (c === "")
 | |
|         {TEX.Error(["MissingArgFor","Missing argument for %1",name])}
 | |
|       if (c === "{") {this.i++} else {this.string = c+"}"+this.string.slice(this.i+1); this.i = 0}
 | |
|       var array = STACKITEM.array().With({
 | |
|         requireClose: true,
 | |
|         arraydef: {
 | |
|           rowspacing: (vspacing||"4pt"),
 | |
|           columnspacing: (spacing||"1em")
 | |
|         }
 | |
|       });
 | |
|       if (cases)         {array.isCases = true}
 | |
|       if (numbered)      {array.isNumbered = true; array.arraydef.side = numbered}
 | |
|       if (open || close) {array.open = open; array.close = close}
 | |
|       if (style === "D") {array.arraydef.displaystyle = true}
 | |
|       if (align != null) {array.arraydef.columnalign = align}
 | |
|       this.Push(array);
 | |
|     },
 | |
|     
 | |
|     Entry: function (name) {
 | |
|       this.Push(STACKITEM.cell().With({isEntry: true, name: name}));
 | |
|       if (this.stack.Top().isCases) {
 | |
|         var string = this.string;
 | |
|         var braces = 0, i = this.i, m = string.length;
 | |
|         while (i < m) {
 | |
|           var c = string.charAt(i);
 | |
|           if (c === "{") {braces++; i++}
 | |
|           else if (c === "}") {if (braces === 0) {m = 0} else {braces--; i++}}
 | |
|           else if (c === "&" && braces === 0) {
 | |
|             TEX.Error(["ExtraAlignTab","Extra alignment tab in \\cases text"]);
 | |
|           } else if (c === "\\") {
 | |
|             if (string.substr(i).match(/^((\\cr)[^a-zA-Z]|\\\\)/)) {m = 0} else {i += 2}
 | |
|           } else {i++}
 | |
|         }
 | |
|         var text = string.substr(this.i,i-this.i);
 | |
|         if (!text.match(/^\s*\\text[^a-zA-Z]/)) {
 | |
|           this.Push.apply(this,this.InternalMath(text,0));
 | |
|           this.i = i;
 | |
|         }
 | |
|       }
 | |
|     },
 | |
|     
 | |
|     Cr: function (name) {
 | |
|       this.Push(STACKITEM.cell().With({isCR: true, name: name}));
 | |
|     },
 | |
|     
 | |
|     CrLaTeX: function (name) {
 | |
|       var n;
 | |
|       if (this.string.charAt(this.i) === "[") {
 | |
|         n = this.GetBrackets(name,"").replace(/ /g,"").replace(/,/,".");
 | |
|         if (n && !this.matchDimen(n)) {
 | |
|           TEX.Error(["BracketMustBeDimension",
 | |
|                      "Bracket argument to %1 must be a dimension",name]);
 | |
|         }
 | |
|       }
 | |
|       this.Push(STACKITEM.cell().With({isCR: true, name: name, linebreak: true}));
 | |
|       var top = this.stack.Top();
 | |
|       if (top.isa(STACKITEM.array)) {
 | |
|         if (n && top.arraydef.rowspacing) {
 | |
|           var rows = top.arraydef.rowspacing.split(/ /);
 | |
|           if (!top.rowspacing) {top.rowspacing = this.dimen2em(rows[0])}
 | |
|           while (rows.length < top.table.length) {rows.push(this.Em(top.rowspacing))}
 | |
|           rows[top.table.length-1] = this.Em(Math.max(0,top.rowspacing+this.dimen2em(n)));
 | |
|           top.arraydef.rowspacing = rows.join(' ');
 | |
|         }
 | |
|       } else {
 | |
|         if (n) {this.Push(MML.mspace().With({depth:n}))}
 | |
|         this.Push(MML.mspace().With({linebreak:MML.LINEBREAK.NEWLINE}));
 | |
|       }
 | |
|     },
 | |
|     emPerInch: 7.2,
 | |
|     pxPerInch: 72,
 | |
|     matchDimen: function (dim) {
 | |
|       return dim.match(/^(-?(?:\.\d+|\d+(?:\.\d*)?))(px|pt|em|ex|mu|pc|in|mm|cm)$/);
 | |
|     },
 | |
|     dimen2em: function (dim) {
 | |
|       var match = this.matchDimen(dim);
 | |
|       var m = parseFloat(match[1]||"1"), unit = match[2];
 | |
|       if (unit === "em") {return m}
 | |
|       if (unit === "ex") {return m * .43}
 | |
|       if (unit === "pt") {return m / 10}                    // 10 pt to an em
 | |
|       if (unit === "pc") {return m * 1.2}                   // 12 pt to a pc
 | |
|       if (unit === "px") {return m * this.emPerInch / this.pxPerInch}
 | |
|       if (unit === "in") {return m * this.emPerInch}
 | |
|       if (unit === "cm") {return m * this.emPerInch / 2.54} // 2.54 cm to an inch
 | |
|       if (unit === "mm") {return m * this.emPerInch / 25.4} // 10 mm to a cm
 | |
|       if (unit === "mu") {return m / 18}
 | |
|       return 0;
 | |
|     },
 | |
|     Em: function (m) {
 | |
|       if (Math.abs(m) < .0006) {return "0em"}
 | |
|       return m.toFixed(3).replace(/\.?0+$/,"") + "em";
 | |
|     },
 | |
|     
 | |
|     HLine: function (name,style) {
 | |
|       if (style == null) {style = "solid"}
 | |
|       var top = this.stack.Top();
 | |
|       if (!top.isa(STACKITEM.array) || top.data.length)
 | |
|         {TEX.Error(["Misplaced","Misplaced %1",name])}
 | |
|       if (top.table.length == 0) {
 | |
|         top.frame.push("top");
 | |
|       } else {
 | |
|         var lines = (top.arraydef.rowlines ? top.arraydef.rowlines.split(/ /) : []);
 | |
|         while (lines.length < top.table.length) {lines.push("none")}
 | |
|         lines[top.table.length-1] = style;
 | |
|         top.arraydef.rowlines = lines.join(' ');
 | |
|       }
 | |
|     },
 | |
|     
 | |
|     HFill: function (name) {
 | |
|       var top = this.stack.Top();
 | |
|       if (top.isa(STACKITEM.array)) top.hfill.push(top.data.length);
 | |
|         else TEX.Error(["UnsupportedHFill","Unsupported use of %1",name]);
 | |
|     },
 | |
|     
 | |
| 
 | |
|     
 | |
|    /************************************************************************/
 | |
|    /*
 | |
|     *   LaTeX environments
 | |
|     */
 | |
| 
 | |
|     BeginEnd: function (name) {
 | |
|       var env = this.GetArgument(name), isEnd = false;
 | |
|       if (env.match(/^\\end\\/)) {isEnd = true; env = env.substr(5)} // special \end{} for \newenvironment environments
 | |
|       if (env.match(/\\/i)) {TEX.Error(["InvalidEnv","Invalid environment name '%1'",env])}
 | |
|       var cmd = this.envFindName(env);
 | |
|       if (!cmd) {TEX.Error(["UnknownEnv","Unknown environment '%1'",env])}
 | |
|       if (!(cmd instanceof Array)) {cmd = [cmd]}
 | |
|       var end = (cmd[1] instanceof Array ? cmd[1][0] : cmd[1]);
 | |
|       var mml = STACKITEM.begin().With({name: env, end: end, parse:this});
 | |
|       if (name === "\\end") {
 | |
|         if (!isEnd && cmd[1] instanceof Array && this[cmd[1][1]]) {
 | |
|           mml = this[cmd[1][1]].apply(this,[mml].concat(cmd.slice(2)));
 | |
|         } else {
 | |
|           mml = STACKITEM.end().With({name: env});
 | |
|         }
 | |
|       } else {
 | |
|         if (++this.macroCount > TEX.config.MAXMACROS) {
 | |
|           TEX.Error(["MaxMacroSub2",
 | |
|                      "MathJax maximum substitution count exceeded; " +
 | |
|                      "is there a recursive latex environment?"]);
 | |
|         }
 | |
|         if (cmd[0] && this[cmd[0]]) {mml = this[cmd[0]].apply(this,[mml].concat(cmd.slice(2)))}
 | |
|       }
 | |
|       this.Push(mml);
 | |
|     },
 | |
|     envFindName: function (name) {return TEXDEF.environment[name]},
 | |
|     
 | |
|     Equation: function (begin,row) {return row},
 | |
|     
 | |
|     ExtensionEnv: function (begin,file) {this.Extension(begin.name,file,"environment")},
 | |
|     
 | |
|     Array: function (begin,open,close,align,spacing,vspacing,style,raggedHeight) {
 | |
|       if (!align) {align = this.GetArgument("\\begin{"+begin.name+"}")}
 | |
|       var 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');
 | |
|       var array = STACKITEM.array().With({
 | |
|         arraydef: {
 | |
|           columnalign: align,
 | |
|           columnspacing: (spacing||"1em"),
 | |
|           rowspacing: (vspacing||"4pt")
 | |
|         }
 | |
|       });
 | |
|       if (lines.match(/[|:]/)) {
 | |
|         if (lines.charAt(0).match(/[|:]/)) {array.frame.push("left"); array.frame.dashed = lines.charAt(0) === ":"}
 | |
|         if (lines.charAt(lines.length-1).match(/[|:]/)) {array.frame.push("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)  {array.open  = this.convertDelimiter(open)}
 | |
|       if (close) {array.close = this.convertDelimiter(close)}
 | |
|       if (style === "D") {array.arraydef.displaystyle = true}
 | |
|          else if (style) {array.arraydef.displaystyle = false}
 | |
|       if (style === "S") {array.arraydef.scriptlevel = 1} // FIXME: should use mstyle?
 | |
|       if (raggedHeight)  {array.arraydef.useHeight = false}
 | |
|       this.Push(begin);
 | |
|       return array;
 | |
|     },
 | |
|     
 | |
|     AlignedArray: function (begin) {
 | |
|       var align = this.GetBrackets("\\begin{"+begin.name+"}");
 | |
|       return this.setArrayAlign(this.Array.apply(this,arguments),align);
 | |
|     },
 | |
|     setArrayAlign: function (array,align) {
 | |
|       align = this.trimSpaces(align||"");
 | |
|       if (align === "t") {array.arraydef.align = "baseline 1"}
 | |
|       else if (align === "b") {array.arraydef.align = "baseline -1"}
 | |
|       else if (align === "c") {array.arraydef.align = "center"}
 | |
|       else if (align) {array.arraydef.align = align} // FIXME: should be an error?
 | |
|       return array;
 | |
|     },
 | |
|     
 | |
|     /************************************************************************/
 | |
|     /*
 | |
|      *   String handling routines
 | |
|      */
 | |
| 
 | |
|     /*
 | |
|      *  Convert delimiter to character
 | |
|      */
 | |
|     convertDelimiter: function (c) {
 | |
|       if (c) {c = TEXDEF.delimiter[c]}
 | |
|       if (c == null) {return null}
 | |
|       if (c instanceof Array) {c = c[0]}
 | |
|       if (c.length === 4) {c = String.fromCharCode(parseInt(c,16))}
 | |
|       return c;
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Trim spaces from a string
 | |
|      */
 | |
|     trimSpaces: function (text) {
 | |
|       if (typeof(text) != 'string') {return text}
 | |
|       return text.replace(/^\s+|\s+$/g,'');
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *   Check if the next character is a space
 | |
|      */
 | |
|     nextIsSpace: function () {
 | |
|       return this.string.charAt(this.i).match(/\s/);
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Get the next non-space character
 | |
|      */
 | |
|     GetNext: function () {
 | |
|       while (this.nextIsSpace()) {this.i++}
 | |
|       return this.string.charAt(this.i);
 | |
|     },
 | |
|   
 | |
|     /*
 | |
|      *  Get and return a control-sequence name
 | |
|      */
 | |
|     GetCS: function () {
 | |
|       var CS = this.string.slice(this.i).match(/^([a-z]+|.) ?/i);
 | |
|       if (CS) {this.i += CS[1].length; return CS[1]} else {this.i++; return " "}
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Get and return a TeX argument (either a single character or control sequence,
 | |
|      *  or the contents of the next set of braces).
 | |
|      */
 | |
|     GetArgument: function (name,noneOK) {
 | |
|       switch (this.GetNext()) {
 | |
|        case "":
 | |
|         if (!noneOK) {TEX.Error(["MissingArgFor","Missing argument for %1",name])}
 | |
|         return null;
 | |
|        case '}':
 | |
|         if (!noneOK) {
 | |
|           TEX.Error(["ExtraCloseMissingOpen",
 | |
|                      "Extra close brace or missing open brace"]);
 | |
|         }
 | |
|         return null;
 | |
|        case '\\':
 | |
|         this.i++; return "\\"+this.GetCS();
 | |
|        case '{':
 | |
|         var j = ++this.i, parens = 1;
 | |
|         while (this.i < this.string.length) {
 | |
|           switch (this.string.charAt(this.i++)) {
 | |
|            case '\\':  this.i++; break;
 | |
|            case '{':   parens++; break;
 | |
|            case '}':
 | |
|             if (--parens == 0) {return this.string.slice(j,this.i-1)}
 | |
|             break;
 | |
|           }
 | |
|         }
 | |
|         TEX.Error(["MissingCloseBrace","Missing close brace"]);
 | |
|         break;
 | |
|       }        
 | |
|       return this.string.charAt(this.i++);
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Get an optional LaTeX argument in brackets
 | |
|      */
 | |
|     GetBrackets: function (name,def) {
 | |
|       if (this.GetNext() != '[') {return def};
 | |
|       var j = ++this.i, parens = 0;
 | |
|       while (this.i < this.string.length) {
 | |
|         switch (this.string.charAt(this.i++)) {
 | |
|          case '{':   parens++; break;
 | |
|          case '\\':  this.i++; break;
 | |
|          case '}':
 | |
|           if (parens-- <= 0) {
 | |
|             TEX.Error(["ExtraCloseLooking",
 | |
|                        "Extra close brace while looking for %1","']'"]);
 | |
|           }
 | |
|           break;   
 | |
|          case ']':
 | |
|           if (parens == 0) {return this.string.slice(j,this.i-1)}
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|       TEX.Error(["MissingCloseBracket",
 | |
|                  "Couldn't find closing ']' for argument to %1",name]);
 | |
|     },
 | |
|   
 | |
|     /*
 | |
|      *  Get the name of a delimiter (check it in the delimiter list).
 | |
|      */
 | |
|     GetDelimiter: function (name,braceOK) {
 | |
|       while (this.nextIsSpace()) {this.i++}
 | |
|       var c = this.string.charAt(this.i); this.i++;
 | |
|       if (this.i <= this.string.length) {
 | |
|         if (c == "\\") {c += this.GetCS(name)}
 | |
|         else if (c === "{" && braceOK) {this.i--; c = this.GetArgument(name)}
 | |
|         if (TEXDEF.delimiter[c] != null) {return this.convertDelimiter(c)}
 | |
|       }
 | |
|       TEX.Error(["MissingOrUnrecognizedDelim",
 | |
|                  "Missing or unrecognized delimiter for %1",name]);
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Get a dimension (including its units).
 | |
|      */
 | |
|     GetDimen: function (name) {
 | |
|       var dimen;
 | |
|       if (this.nextIsSpace()) {this.i++}
 | |
|       if (this.string.charAt(this.i) == '{') {
 | |
|         dimen = this.GetArgument(name);
 | |
|         if (dimen.match(/^\s*([-+]?([.,]\d+|\d+([.,]\d*)?))\s*(pt|em|ex|mu|px|mm|cm|in|pc)\s*$/))
 | |
|           {return dimen.replace(/ /g,"").replace(/,/,".")}
 | |
|       } else {
 | |
|         dimen = this.string.slice(this.i);
 | |
|         var match = dimen.match(/^\s*(([-+]?([.,]\d+|\d+([.,]\d*)?))\s*(pt|em|ex|mu|px|mm|cm|in|pc)) ?/);
 | |
|         if (match) {
 | |
|           this.i += match[0].length;
 | |
|           return match[1].replace(/ /g,"").replace(/,/,".");
 | |
|         }
 | |
|       }
 | |
|       TEX.Error(["MissingDimOrUnits",
 | |
|                  "Missing dimension or its units for %1",name]);
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Get everything up to the given control sequence (token)
 | |
|      */
 | |
|     GetUpTo: function (name,token) {
 | |
|       while (this.nextIsSpace()) {this.i++}
 | |
|       var j = this.i, k, c, parens = 0;
 | |
|       while (this.i < this.string.length) {
 | |
|         k = this.i; c = this.string.charAt(this.i++);
 | |
|         switch (c) {
 | |
|          case '\\':  c += this.GetCS(); break;
 | |
|          case '{':   parens++; break;
 | |
|          case '}':
 | |
|           if (parens == 0) {
 | |
|             TEX.Error(["ExtraCloseLooking",
 | |
|                        "Extra close brace while looking for %1",token])
 | |
|           }
 | |
|           parens--;
 | |
|           break;
 | |
|         }
 | |
|         if (parens == 0 && c == token) {return this.string.slice(j,k)}
 | |
|       }
 | |
|       TEX.Error(["TokenNotFoundForCommand",
 | |
|                  "Couldn't find %1 for %2",token,name]);
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Parse various substrings
 | |
|      */
 | |
|     ParseArg: function (name) {return TEX.Parse(this.GetArgument(name),this.stack.env).mml()},
 | |
|     ParseUpTo: function (name,token) {return TEX.Parse(this.GetUpTo(name,token),this.stack.env).mml()},
 | |
|     
 | |
|     /*
 | |
|      *  Break up a string into text and math blocks
 | |
|      */
 | |
|     InternalMath: function (text,level) {
 | |
|       var def = (this.stack.env.font ? {mathvariant: this.stack.env.font} : {});
 | |
|       var mml = [], i = 0, k = 0, c, match = '', braces = 0;
 | |
|       if (text.match(/\\?[${}\\]|\\\(|\\(eq)?ref\s*\{/)) {
 | |
|         while (i < text.length) {
 | |
|           c = text.charAt(i++);
 | |
|           if (c === '$') {
 | |
|             if (match === '$' && braces === 0) {
 | |
|               mml.push(MML.TeXAtom(TEX.Parse(text.slice(k,i-1),{}).mml()));
 | |
|               match = ''; k = i;
 | |
|             } else if (match === '') {
 | |
|               if (k < i-1) mml.push(this.InternalText(text.slice(k,i-1),def));
 | |
|               match = '$'; k = i;
 | |
|             }
 | |
|           } else if (c === '{' && match !== '') {
 | |
|             braces++;
 | |
|           } else if (c === '}') {
 | |
|             if (match === '}' && braces === 0) {
 | |
|               mml.push(MML.TeXAtom(TEX.Parse(text.slice(k,i),{}).mml().With(def)));
 | |
|               match = ''; k = i;
 | |
|             } else if (match !== '') {
 | |
|               if (braces) braces--;
 | |
|             }
 | |
|           } else if (c === '\\') {
 | |
|             if (match === '' && text.substr(i).match(/^(eq)?ref\s*\{/)) {
 | |
|               var len = RegExp["$&"].length;
 | |
|               if (k < i-1) mml.push(this.InternalText(text.slice(k,i-1),def));
 | |
|               match = '}'; k = i-1; i += len;
 | |
|             } else {
 | |
|               c = text.charAt(i++);
 | |
|               if (c === '(' && match === '') {
 | |
|                 if (k < i-2) mml.push(this.InternalText(text.slice(k,i-2),def));
 | |
|                 match = ')'; k = i;
 | |
|               } else if (c === ')' && match === ')' && braces === 0) {
 | |
|                 mml.push(MML.TeXAtom(TEX.Parse(text.slice(k,i-2),{}).mml()));
 | |
|                 match = ''; k = i;
 | |
|               } else if (c.match(/[${}\\]/) && match === '')  {
 | |
|                 i--; text = text.substr(0,i-1) + text.substr(i); // remove \ from \$, \{, \}, or \\
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|         if (match !== '') TEX.Error(["MathNotTerminated","Math not terminated in text box"]);
 | |
|       }
 | |
|       if (k < text.length) mml.push(this.InternalText(text.slice(k),def));
 | |
|       if (level != null) {
 | |
|         mml = [MML.mstyle.apply(MML,mml).With({displaystyle:false,scriptlevel:level})];
 | |
|       } else if (mml.length > 1) {
 | |
|         mml = [MML.mrow.apply(MML,mml)];
 | |
|       }
 | |
|       return mml;
 | |
|     },
 | |
|     InternalText: function (text,def) {
 | |
|       text = text.replace(/^\s+/,NBSP).replace(/\s+$/,NBSP);
 | |
|       return MML.mtext(MML.chars(text)).With(def);
 | |
|     },
 | |
| 
 | |
|     /*
 | |
|      *  Replace macro paramters with their values
 | |
|      */
 | |
|     SubstituteArgs: function (args,string) {
 | |
|       var text = ''; var newstring = ''; var c; var i = 0;
 | |
|       while (i < string.length) {
 | |
|         c = string.charAt(i++);
 | |
|         if (c === "\\") {text += c + string.charAt(i++)}
 | |
|         else if (c === '#') {
 | |
|           c = string.charAt(i++);
 | |
|           if (c === '#') {text += c} else {
 | |
|             if (!c.match(/[1-9]/) || c > args.length) {
 | |
|               TEX.Error(["IllegalMacroParam",
 | |
|                          "Illegal macro parameter reference"]);
 | |
|             }
 | |
|             newstring = this.AddArgs(this.AddArgs(newstring,text),args[c-1]);
 | |
|             text = '';
 | |
|           }
 | |
|         } else {text += c}
 | |
|       }
 | |
|       return this.AddArgs(newstring,text);
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Make sure that macros are followed by a space if their names
 | |
|      *  could accidentally be continued into the following text.
 | |
|      */
 | |
|     AddArgs: function (s1,s2) {
 | |
|       if (s2.match(/^[a-z]/i) && s1.match(/(^|[^\\])(\\\\)*\\[a-z]+$/i)) {s1 += ' '}
 | |
|       if (s1.length + s2.length > TEX.config.MAXBUFFER) {
 | |
|         TEX.Error(["MaxBufferSize",
 | |
|                    "MathJax internal buffer size exceeded; is there a recursive macro call?"]);
 | |
|       }
 | |
|       return s1+s2;
 | |
|     }
 | |
|     
 | |
|   });
 | |
|   
 | |
|   /************************************************************************/
 | |
| 
 | |
|   TEX.Augment({
 | |
|     Stack: STACK, Parse: PARSE, Definitions: TEXDEF, Startup: STARTUP,
 | |
|     
 | |
|     config: {
 | |
|       MAXMACROS: 10000,    // maximum number of macro substitutions per equation
 | |
|       MAXBUFFER: 5*1024    // maximum size of TeX string to process
 | |
|     },
 | |
|     
 | |
|     sourceMenuTitle: /*_(MathMenu)*/ ["TeXCommands","TeX Commands"],
 | |
|     annotationEncoding: "application/x-tex",
 | |
| 
 | |
|     prefilterHooks: MathJax.Callback.Hooks(true),    // hooks to run before processing TeX
 | |
|     postfilterHooks: MathJax.Callback.Hooks(true),   // hooks to run after processing TeX
 | |
|     
 | |
|     //
 | |
|     //  Check if AMSmath extension must be loaded and push
 | |
|     //    it on the extensions array, if needed
 | |
|     //
 | |
|     Config: function () {
 | |
|       this.SUPER(arguments).Config.apply(this,arguments);
 | |
|       if (this.config.equationNumbers.autoNumber !== "none") {
 | |
|         if (!this.config.extensions) {this.config.extensions = []}
 | |
|         this.config.extensions.push("AMSmath.js");
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     //
 | |
|     //  Convert TeX to ElementJax
 | |
|     //
 | |
|     Translate: function (script) {
 | |
|       var mml, isError = false, math = MathJax.HTML.getScript(script);
 | |
|       var display = (script.type.replace(/\n/g," ").match(/(;|\s|\n)mode\s*=\s*display(;|\s|\n|$)/) != null);
 | |
|       var data = {math:math, display:display, script:script};
 | |
|       var callback = this.prefilterHooks.Execute(data); if (callback) return callback;
 | |
|       math = data.math;
 | |
|       try {
 | |
|         mml = TEX.Parse(math).mml();
 | |
|       } catch(err) {
 | |
|         if (!err.texError) {throw err}
 | |
|         mml = this.formatError(err,math,display,script);
 | |
|         isError = true;
 | |
|       }
 | |
|       if (mml.isa(MML.mtable) && mml.displaystyle === "inherit") mml.displaystyle = display; // for tagged equations
 | |
|       if (mml.inferred) {mml = MML.apply(MathJax.ElementJax,mml.data)} else {mml = MML(mml)}
 | |
|       if (display) {mml.root.display = "block"}
 | |
|       if (isError) {mml.texError = true}
 | |
|       data.math = mml; 
 | |
|       return this.postfilterHooks.Execute(data) || data.math;
 | |
|     },
 | |
|     prefilterMath: function (math,displaystyle,script) {
 | |
|       return math;
 | |
|     },
 | |
|     postfilterMath: function (math,displaystyle,script) {
 | |
|       this.combineRelations(math.root);
 | |
|       return math;
 | |
|     },
 | |
|     formatError: function (err,math,display,script) {
 | |
|       var message = err.message.replace(/\n.*/,"");
 | |
|       HUB.signal.Post(["TeX Jax - parse error",message,math,display,script]);
 | |
|       return MML.Error(message);
 | |
|     },
 | |
| 
 | |
|     //
 | |
|     //  Produce an error and stop processing this equation
 | |
|     //
 | |
|     Error: function (message) {
 | |
|       //
 | |
|       //  Translate message if it is ["id","message",args]
 | |
|       //
 | |
|       if (message instanceof Array) {message = _.apply(_,message)}
 | |
|       throw HUB.Insert(Error(message),{texError: true});
 | |
|     },
 | |
|     
 | |
|     //
 | |
|     //  Add a user-defined macro to the macro list
 | |
|     //
 | |
|     Macro: function (name,def,argn) {
 | |
|       TEXDEF.macros[name] = ['Macro'].concat([].slice.call(arguments,1));
 | |
|       TEXDEF.macros[name].isUser = true;
 | |
|     },
 | |
|     
 | |
|     /*
 | |
|      *  Create an mrow that has stretchy delimiters at either end, as needed
 | |
|      */
 | |
|     fenced: function (open,mml,close) {
 | |
|       var mrow = MML.mrow().With({open:open, close:close, texClass:MML.TEXCLASS.INNER});
 | |
|       mrow.Append(MML.mo(open).With({fence:true, stretchy:true, texClass:MML.TEXCLASS.OPEN}));
 | |
|       if (mml.type === "mrow") {mrow.Append.apply(mrow,mml.data)} else {mrow.Append(mml)}
 | |
|       mrow.Append(MML.mo(close).With({fence:true, stretchy:true, texClass:MML.TEXCLASS.CLOSE}));
 | |
|       return mrow;
 | |
|     },
 | |
|     /*
 | |
|      *  Create an mrow that has \mathchoice using \bigg and \big for the delimiters
 | |
|      */
 | |
|     fixedFence: function (open,mml,close) {
 | |
|       var mrow = MML.mrow().With({open:open, close:close, texClass:MML.TEXCLASS.ORD});
 | |
|       if (open) {mrow.Append(this.mathPalette(open,"l"))}
 | |
|       if (mml.type === "mrow") {mrow.Append.apply(mrow,mml.data)} else {mrow.Append(mml)}
 | |
|       if (close) {mrow.Append(this.mathPalette(close,"r"))}
 | |
|       return mrow;
 | |
|     },
 | |
|     mathPalette: function (fence,side) {
 | |
|       if (fence === '{' || fence === '}') {fence = "\\"+fence}
 | |
|       var D = '{\\bigg'+side+' '+fence+'}', T = '{\\big'+side+' '+fence+'}';
 | |
|       return TEX.Parse('\\mathchoice'+D+T+T+T,{}).mml();
 | |
|     },
 | |
|     
 | |
|     //
 | |
|     //  Combine adjacent <mo> elements that are relations
 | |
|     //    (since MathML treats the spacing very differently)
 | |
|     //
 | |
|     combineRelations: function (mml) {
 | |
|       var i, m, m1, m2;
 | |
|       for (i = 0, m = mml.data.length; i < m; i++) {
 | |
|         if (mml.data[i]) {
 | |
|           if (mml.isa(MML.mrow)) {
 | |
|             while (i+1 < m && (m1 = mml.data[i]) && (m2 = mml.data[i+1]) &&
 | |
|                    m1.isa(MML.mo) && m2.isa(MML.mo) &&
 | |
|                    m1.Get("texClass") === MML.TEXCLASS.REL &&
 | |
|                    m2.Get("texClass") === MML.TEXCLASS.REL) {
 | |
|               if (m1.variantForm == m2.variantForm &&
 | |
|                   m1.Get("mathvariant") == m2.Get("mathvariant") && m1.style == m2.style &&
 | |
|                   m1["class"] == m2["class"] && !m1.id && !m2.id) {
 | |
|                 m1.Append.apply(m1,m2.data);
 | |
|                 mml.data.splice(i+1,1); m--;
 | |
|               } else {
 | |
|                 m1.rspace = m2.lspace = "0pt"; i++;
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|           if (!mml.data[i].isToken) {this.combineRelations(mml.data[i])}
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   //
 | |
|   //  Add the default filters
 | |
|   //
 | |
|   TEX.prefilterHooks.Add(function (data) {
 | |
|     data.math = TEX.prefilterMath(data.math,data.display,data.script);
 | |
|   });
 | |
|   TEX.postfilterHooks.Add(function (data) {
 | |
|     data.math = TEX.postfilterMath(data.math,data.display,data.script);
 | |
|   });
 | |
| 
 | |
|   TEX.loadComplete("jax.js");
 | |
|   
 | |
| })(MathJax.InputJax.TeX,MathJax.Hub,MathJax.Ajax);
 |