/* -*- Mode: Javascript; indent-tabs-mode:nil; js-indent-level: 2 -*- */ /* vim: set ts=2 et sw=2 tw=80: */ /************************************************************* * * MathJax/extensions/toMathML.js * * Implements a toMathML() method for the mml Element Jax that returns * a MathML string from a given math expression. * * --------------------------------------------------------------------- * * Copyright (c) 2010-2013 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. */ MathJax.Hub.Register.LoadHook("[MathJax]/jax/element/mml/jax.js",function () { var VERSION = "2.3"; var MML = MathJax.ElementJax.mml SETTINGS = MathJax.Hub.config.menuSettings; MML.mbase.Augment({ toMathML: function (space) { var inferred = (this.inferred && this.parent.inferRow); if (space == null) {space = ""} var tag = this.type, attr = this.toMathMLattributes(); if (tag === "mspace") {return space + "<"+tag+attr+" />"} var data = []; var SPACE = (this.isToken ? "" : space+(inferred ? "" : " ")); for (var i = 0, m = this.data.length; i < m; i++) { if (this.data[i]) {data.push(this.data[i].toMathML(SPACE))} else if (!this.isToken) {data.push(SPACE+"")} } if (this.isToken) {return space + "<"+tag+attr+">"+data.join("")+""} if (inferred) {return data.join("\n")} if (data.length === 0 || (data.length === 1 && data[0] === "")) {return space + "<"+tag+attr+" />"} return space + "<"+tag+attr+">\n"+data.join("\n")+"\n"+ space +""; }, toMathMLattributes: function () { var attr = [], defaults = this.defaults; var copy = (this.attrNames||MML.copyAttributeNames), skip = MML.skipAttributes; if (this.type === "math" && (!this.attr || !this.attr.xmlns)) {attr.push('xmlns="http://www.w3.org/1998/Math/MathML"')} if (!this.attrNames) { if (this.type === "mstyle") {defaults = MML.math.prototype.defaults} for (var id in defaults) {if (!skip[id] && defaults.hasOwnProperty(id)) { var force = (id === "open" || id === "close"); if (this[id] != null && (force || this[id] !== defaults[id])) { var value = this[id]; delete this[id]; if (force || this.Get(id) !== value) {attr.push(id+'="'+this.toMathMLattribute(value)+'"')} this[id] = value; } }} } for (var i = 0, m = copy.length; i < m; i++) { if (copy[i] === "class") continue; // this is handled separately below value = (this.attr||{})[copy[i]]; if (value == null) {value = this[copy[i]]} if (value != null) {attr.push(copy[i]+'="'+this.toMathMLquote(value)+'"')} } this.toMathMLclass(attr); if (attr.length) {return " "+attr.join(" ")} else {return ""} }, toMathMLclass: function (attr) { var CLASS = []; if (this["class"]) {CLASS.push(this["class"])} if (this.isa(MML.TeXAtom) && SETTINGS.texHints) { var TEXCLASS = ["ORD","OP","BIN","REL","OPEN","CLOSE","PUNCT","INNER","VCENTER"][this.texClass]; if (TEXCLASS) {CLASS.push("MJX-TeXAtom-"+TEXCLASS)} } if (this.mathvariant && this.toMathMLvariants[this.mathvariant]) {CLASS.push("MJX"+this.mathvariant)} if (this.variantForm) {CLASS.push("MJX-variant")} if (CLASS.length) {attr.unshift('class="'+CLASS.join(" ")+'"')} }, toMathMLattribute: function (value) { if (typeof(value) === "string" && value.replace(/ /g,"").match(/^(([-+])?(\d+(\.\d*)?|\.\d+))mu$/)) { // FIXME: should take scriptlevel into account return RegExp.$2+((1/18)*RegExp.$3).toFixed(3).replace(/\.?0+$/,"")+"em"; } else if (this.toMathMLvariants[value]) {return this.toMathMLvariants[value]} return this.toMathMLquote(value); }, toMathMLvariants: { "-tex-caligraphic": MML.VARIANT.SCRIPT, "-tex-caligraphic-bold": MML.VARIANT.BOLDSCRIPT, "-tex-oldstyle": MML.VARIANT.NORMAL, "-tex-oldstyle-bold": MML.VARIANT.BOLD, "-tex-mathit": MML.VARIANT.ITALIC }, toMathMLquote: function (string) { string = String(string).split(""); for (var i = 0, m = string.length; i < m; i++) { var n = string[i].charCodeAt(0); if (n <= 0xD7FF || 0xE000 <= n) { // Code points U+0000 to U+D7FF and U+E000 to U+FFFF. // They are directly represented by n. if (n < 0x20 || n > 0x7E) { string[i] = "&#x"+n.toString(16).toUpperCase()+";"; } else { var c = {'&':'&', '<':'<', '>':'>', '"':'"'}[string[i]]; if (c) {string[i] = c} } } else if (i+1 < m) { // Code points U+10000 to U+10FFFF. // n is the lead surrogate, let's read the trail surrogate. var trailSurrogate = string[i+1].charCodeAt(0); var codePoint = (((n-0xD800)<<10)+(trailSurrogate-0xDC00)+0x10000); string[i] = "&#x"+codePoint.toString(16).toUpperCase()+";"; string[i+1] = ""; i++; } else { // n is a lead surrogate without corresponding trail surrogate: // remove that character. string[i] = ""; } } return string.join(""); } }); MML.msubsup.Augment({ toMathML: function (space) { var tag = this.type; if (this.data[this.sup] == null) {tag = "msub"} if (this.data[this.sub] == null) {tag = "msup"} var attr = this.toMathMLattributes(); delete this.data[0].inferred; var data = []; for (var i = 0, m = this.data.length; i < m; i++) {if (this.data[i]) {data.push(this.data[i].toMathML(space+" "))}} return space + "<"+tag+attr+">\n" + data.join("\n") + "\n" + space + ""; } }); MML.munderover.Augment({ toMathML: function (space) { var tag = this.type; if (this.data[this.under] == null) {tag = "mover"} if (this.data[this.over] == null) {tag = "munder"} var attr = this.toMathMLattributes(); delete this.data[0].inferred; var data = []; for (var i = 0, m = this.data.length; i < m; i++) {if (this.data[i]) {data.push(this.data[i].toMathML(space+" "))}} return space + "<"+tag+attr+">\n" + data.join("\n") + "\n" + space + ""; } }); MML.TeXAtom.Augment({ toMathML: function (space) { // FIXME: Handle spacing using mpadded? var attr = this.toMathMLattributes(); if (!attr && this.data[0].data.length === 1) {return space.substr(2) + this.data[0].toMathML(space)} return space+"\n" + this.data[0].toMathML(space+" ")+"\n"+space+""; } }); MML.chars.Augment({ toMathML: function (space) {return (space||"") + this.toMathMLquote(this.toString())} }); MML.entity.Augment({ toMathML: function (space) {return (space||"") + "&"+this.data[0]+";"} }); MML.xml.Augment({ toMathML: function (space) {return (space||"") + this.toString()} }); MathJax.Hub.Register.StartupHook("TeX mathchoice Ready",function () { MML.TeXmathchoice.Augment({ toMathML: function (space) {return this.Core().toMathML(space)} }); }); MathJax.Hub.Startup.signal.Post("toMathML Ready"); }); MathJax.Ajax.loadComplete("[MathJax]/extensions/toMathML.js");