mirror of
				https://github.com/TriliumNext/Notes.git
				synced 2025-10-27 01:51:34 +08:00 
			
		
		
		
	
		
			
	
	
		
			3110 lines
		
	
	
		
			74 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			3110 lines
		
	
	
		
			74 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|   | import katex from '../katex.mjs'; | ||
|  | 
 | ||
|  | /* eslint-disable */ | ||
|  | 
 | ||
|  | /* -*- Mode: Javascript; indent-tabs-mode:nil; js-indent-level: 2 -*- */ | ||
|  | 
 | ||
|  | /* vim: set ts=2 et sw=2 tw=80: */ | ||
|  | 
 | ||
|  | /************************************************************* | ||
|  |  * | ||
|  |  *  KaTeX mhchem.js | ||
|  |  * | ||
|  |  *  This file implements a KaTeX version of mhchem version 3.3.0. | ||
|  |  *  It is adapted from MathJax/extensions/TeX/mhchem.js | ||
|  |  *  It differs from the MathJax version as follows: | ||
|  |  *    1. The interface is changed so that it can be called from KaTeX, not MathJax. | ||
|  |  *    2. \rlap and \llap are replaced with \mathrlap and \mathllap. | ||
|  |  *    3. Four lines of code are edited in order to use \raisebox instead of \raise. | ||
|  |  *    4. The reaction arrow code is simplified. All reaction arrows are rendered | ||
|  |  *       using KaTeX extensible arrows instead of building non-extensible arrows. | ||
|  |  *    5. \tripledash vertical alignment is slightly adjusted. | ||
|  |  * | ||
|  |  *    This code, as other KaTeX code, is released under the MIT license. | ||
|  |  *  | ||
|  |  * /************************************************************* | ||
|  |  * | ||
|  |  *  MathJax/extensions/TeX/mhchem.js | ||
|  |  * | ||
|  |  *  Implements the \ce command for handling chemical formulas | ||
|  |  *  from the mhchem LaTeX package. | ||
|  |  * | ||
|  |  *  --------------------------------------------------------------------- | ||
|  |  * | ||
|  |  *  Copyright (c) 2011-2015 The MathJax Consortium | ||
|  |  *  Copyright (c) 2015-2018 Martin Hensel | ||
|  |  * | ||
|  |  *  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. | ||
|  |  */ | ||
|  | //
 | ||
|  | // Coding Style
 | ||
|  | //   - use '' for identifiers that can by minified/uglified
 | ||
|  | //   - use "" for strings that need to stay untouched
 | ||
|  | // version: "3.3.0" for MathJax and KaTeX
 | ||
|  | // Add \ce, \pu, and \tripledash to the KaTeX macros.
 | ||
|  | katex.__defineMacro("\\ce", function (context) { | ||
|  |   return chemParse(context.consumeArgs(1)[0], "ce"); | ||
|  | }); | ||
|  | 
 | ||
|  | katex.__defineMacro("\\pu", function (context) { | ||
|  |   return chemParse(context.consumeArgs(1)[0], "pu"); | ||
|  | }); //  Needed for \bond for the ~ forms
 | ||
|  | //  Raise by 2.56mu, not 2mu. We're raising a hyphen-minus, U+002D, not 
 | ||
|  | //  a mathematical minus, U+2212. So we need that extra 0.56.
 | ||
|  | 
 | ||
|  | 
 | ||
|  | katex.__defineMacro("\\tripledash", "{\\vphantom{-}\\raisebox{2.56mu}{$\\mkern2mu" + "\\tiny\\text{-}\\mkern1mu\\text{-}\\mkern1mu\\text{-}\\mkern2mu$}}"); | ||
|  | //  This is the main function for handing the \ce and \pu commands.
 | ||
|  | //  It takes the argument to \ce or \pu and returns the corresponding TeX string.
 | ||
|  | //
 | ||
|  | 
 | ||
|  | var chemParse = function chemParse(tokens, stateMachine) { | ||
|  |   // Recreate the argument string from KaTeX's array of tokens.
 | ||
|  |   var str = ""; | ||
|  |   var expectedLoc = tokens[tokens.length - 1].loc.start; | ||
|  | 
 | ||
|  |   for (var i = tokens.length - 1; i >= 0; i--) { | ||
|  |     if (tokens[i].loc.start > expectedLoc) { | ||
|  |       // context.consumeArgs has eaten a space.
 | ||
|  |       str += " "; | ||
|  |       expectedLoc = tokens[i].loc.start; | ||
|  |     } | ||
|  | 
 | ||
|  |     str += tokens[i].text; | ||
|  |     expectedLoc += tokens[i].text.length; | ||
|  |   } | ||
|  | 
 | ||
|  |   var tex = texify.go(mhchemParser.go(str, stateMachine)); | ||
|  |   return tex; | ||
|  | }; //
 | ||
|  | // Core parser for mhchem syntax  (recursive)
 | ||
|  | //
 | ||
|  | 
 | ||
|  | /** @type {MhchemParser} */ | ||
|  | 
 | ||
|  | 
 | ||
|  | var mhchemParser = { | ||
|  |   //
 | ||
|  |   // Parses mchem \ce syntax
 | ||
|  |   //
 | ||
|  |   // Call like
 | ||
|  |   //   go("H2O");
 | ||
|  |   //
 | ||
|  |   go: function go(input, stateMachine) { | ||
|  |     if (!input) { | ||
|  |       return []; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (stateMachine === undefined) { | ||
|  |       stateMachine = 'ce'; | ||
|  |     } | ||
|  | 
 | ||
|  |     var state = '0'; //
 | ||
|  |     // String buffers for parsing:
 | ||
|  |     //
 | ||
|  |     // buffer.a == amount
 | ||
|  |     // buffer.o == element
 | ||
|  |     // buffer.b == left-side superscript
 | ||
|  |     // buffer.p == left-side subscript
 | ||
|  |     // buffer.q == right-side subscript
 | ||
|  |     // buffer.d == right-side superscript
 | ||
|  |     //
 | ||
|  |     // buffer.r == arrow
 | ||
|  |     // buffer.rdt == arrow, script above, type
 | ||
|  |     // buffer.rd == arrow, script above, content
 | ||
|  |     // buffer.rqt == arrow, script below, type
 | ||
|  |     // buffer.rq == arrow, script below, content
 | ||
|  |     //
 | ||
|  |     // buffer.text_
 | ||
|  |     // buffer.rm
 | ||
|  |     // etc.
 | ||
|  |     //
 | ||
|  |     // buffer.parenthesisLevel == int, starting at 0
 | ||
|  |     // buffer.sb == bool, space before
 | ||
|  |     // buffer.beginsWithBond == bool
 | ||
|  |     //
 | ||
|  |     // These letters are also used as state names.
 | ||
|  |     //
 | ||
|  |     // Other states:
 | ||
|  |     // 0 == begin of main part (arrow/operator unlikely)
 | ||
|  |     // 1 == next entity
 | ||
|  |     // 2 == next entity (arrow/operator unlikely)
 | ||
|  |     // 3 == next atom
 | ||
|  |     // c == macro
 | ||
|  |     //
 | ||
|  | 
 | ||
|  |     /** @type {Buffer} */ | ||
|  | 
 | ||
|  |     var buffer = {}; | ||
|  |     buffer['parenthesisLevel'] = 0; | ||
|  |     input = input.replace(/\n/g, " "); | ||
|  |     input = input.replace(/[\u2212\u2013\u2014\u2010]/g, "-"); | ||
|  |     input = input.replace(/[\u2026]/g, "..."); //
 | ||
|  |     // Looks through mhchemParser.transitions, to execute a matching action
 | ||
|  |     // (recursive)
 | ||
|  |     //
 | ||
|  | 
 | ||
|  |     var lastInput; | ||
|  |     var watchdog = 10; | ||
|  |     /** @type {ParserOutput[]} */ | ||
|  | 
 | ||
|  |     var output = []; | ||
|  | 
 | ||
|  |     while (true) { | ||
|  |       if (lastInput !== input) { | ||
|  |         watchdog = 10; | ||
|  |         lastInput = input; | ||
|  |       } else { | ||
|  |         watchdog--; | ||
|  |       } //
 | ||
|  |       // Find actions in transition table
 | ||
|  |       //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |       var machine = mhchemParser.stateMachines[stateMachine]; | ||
|  |       var t = machine.transitions[state] || machine.transitions['*']; | ||
|  | 
 | ||
|  |       iterateTransitions: for (var i = 0; i < t.length; i++) { | ||
|  |         var matches = mhchemParser.patterns.match_(t[i].pattern, input); | ||
|  | 
 | ||
|  |         if (matches) { | ||
|  |           //
 | ||
|  |           // Execute actions
 | ||
|  |           //
 | ||
|  |           var task = t[i].task; | ||
|  | 
 | ||
|  |           for (var iA = 0; iA < task.action_.length; iA++) { | ||
|  |             var o; //
 | ||
|  |             // Find and execute action
 | ||
|  |             //
 | ||
|  | 
 | ||
|  |             if (machine.actions[task.action_[iA].type_]) { | ||
|  |               o = machine.actions[task.action_[iA].type_](buffer, matches.match_, task.action_[iA].option); | ||
|  |             } else if (mhchemParser.actions[task.action_[iA].type_]) { | ||
|  |               o = mhchemParser.actions[task.action_[iA].type_](buffer, matches.match_, task.action_[iA].option); | ||
|  |             } else { | ||
|  |               throw ["MhchemBugA", "mhchem bug A. Please report. (" + task.action_[iA].type_ + ")"]; // Trying to use non-existing action
 | ||
|  |             } //
 | ||
|  |             // Add output
 | ||
|  |             //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |             mhchemParser.concatArray(output, o); | ||
|  |           } //
 | ||
|  |           // Set next state,
 | ||
|  |           // Shorten input,
 | ||
|  |           // Continue with next character
 | ||
|  |           //   (= apply only one transition per position)
 | ||
|  |           //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |           state = task.nextState || state; | ||
|  | 
 | ||
|  |           if (input.length > 0) { | ||
|  |             if (!task.revisit) { | ||
|  |               input = matches.remainder; | ||
|  |             } | ||
|  | 
 | ||
|  |             if (!task.toContinue) { | ||
|  |               break iterateTransitions; | ||
|  |             } | ||
|  |           } else { | ||
|  |             return output; | ||
|  |           } | ||
|  |         } | ||
|  |       } //
 | ||
|  |       // Prevent infinite loop
 | ||
|  |       //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |       if (watchdog <= 0) { | ||
|  |         throw ["MhchemBugU", "mhchem bug U. Please report."]; // Unexpected character
 | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   concatArray: function concatArray(a, b) { | ||
|  |     if (b) { | ||
|  |       if (Array.isArray(b)) { | ||
|  |         for (var iB = 0; iB < b.length; iB++) { | ||
|  |           a.push(b[iB]); | ||
|  |         } | ||
|  |       } else { | ||
|  |         a.push(b); | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   patterns: { | ||
|  |     //
 | ||
|  |     // Matching patterns
 | ||
|  |     // either regexps or function that return null or {match_:"a", remainder:"bc"}
 | ||
|  |     //
 | ||
|  |     patterns: { | ||
|  |       // property names must not look like integers ("2") for correct property traversal order, later on
 | ||
|  |       'empty': /^$/, | ||
|  |       'else': /^./, | ||
|  |       'else2': /^./, | ||
|  |       'space': /^\s/, | ||
|  |       'space A': /^\s(?=[A-Z\\$])/, | ||
|  |       'space$': /^\s$/, | ||
|  |       'a-z': /^[a-z]/, | ||
|  |       'x': /^x/, | ||
|  |       'x$': /^x$/, | ||
|  |       'i$': /^i$/, | ||
|  |       'letters': /^(?:[a-zA-Z\u03B1-\u03C9\u0391-\u03A9?@]|(?:\\(?:alpha|beta|gamma|delta|epsilon|zeta|eta|theta|iota|kappa|lambda|mu|nu|xi|omicron|pi|rho|sigma|tau|upsilon|phi|chi|psi|omega|Gamma|Delta|Theta|Lambda|Xi|Pi|Sigma|Upsilon|Phi|Psi|Omega)(?:\s+|\{\}|(?![a-zA-Z]))))+/, | ||
|  |       '\\greek': /^\\(?:alpha|beta|gamma|delta|epsilon|zeta|eta|theta|iota|kappa|lambda|mu|nu|xi|omicron|pi|rho|sigma|tau|upsilon|phi|chi|psi|omega|Gamma|Delta|Theta|Lambda|Xi|Pi|Sigma|Upsilon|Phi|Psi|Omega)(?:\s+|\{\}|(?![a-zA-Z]))/, | ||
|  |       'one lowercase latin letter $': /^(?:([a-z])(?:$|[^a-zA-Z]))$/, | ||
|  |       '$one lowercase latin letter$ $': /^\$(?:([a-z])(?:$|[^a-zA-Z]))\$$/, | ||
|  |       'one lowercase greek letter $': /^(?:\$?[\u03B1-\u03C9]\$?|\$?\\(?:alpha|beta|gamma|delta|epsilon|zeta|eta|theta|iota|kappa|lambda|mu|nu|xi|omicron|pi|rho|sigma|tau|upsilon|phi|chi|psi|omega)\s*\$?)(?:\s+|\{\}|(?![a-zA-Z]))$/, | ||
|  |       'digits': /^[0-9]+/, | ||
|  |       '-9.,9': /^[+\-]?(?:[0-9]+(?:[,.][0-9]+)?|[0-9]*(?:\.[0-9]+))/, | ||
|  |       '-9.,9 no missing 0': /^[+\-]?[0-9]+(?:[.,][0-9]+)?/, | ||
|  |       '(-)(9.,9)(e)(99)': function e99(input) { | ||
|  |         var m = input.match(/^(\+\-|\+\/\-|\+|\-|\\pm\s?)?([0-9]+(?:[,.][0-9]+)?|[0-9]*(?:\.[0-9]+))?(\((?:[0-9]+(?:[,.][0-9]+)?|[0-9]*(?:\.[0-9]+))\))?(?:([eE]|\s*(\*|x|\\times|\u00D7)\s*10\^)([+\-]?[0-9]+|\{[+\-]?[0-9]+\}))?/); | ||
|  | 
 | ||
|  |         if (m && m[0]) { | ||
|  |           return { | ||
|  |             match_: m.splice(1), | ||
|  |             remainder: input.substr(m[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }, | ||
|  |       '(-)(9)^(-9)': function _(input) { | ||
|  |         var m = input.match(/^(\+\-|\+\/\-|\+|\-|\\pm\s?)?([0-9]+(?:[,.][0-9]+)?|[0-9]*(?:\.[0-9]+)?)\^([+\-]?[0-9]+|\{[+\-]?[0-9]+\})/); | ||
|  | 
 | ||
|  |         if (m && m[0]) { | ||
|  |           return { | ||
|  |             match_: m.splice(1), | ||
|  |             remainder: input.substr(m[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }, | ||
|  |       'state of aggregation $': function stateOfAggregation$(input) { | ||
|  |         // ... or crystal system
 | ||
|  |         var a = mhchemParser.patterns.findObserveGroups(input, "", /^\([a-z]{1,3}(?=[\),])/, ")", ""); // (aq), (aq,$\infty$), (aq, sat)
 | ||
|  | 
 | ||
|  |         if (a && a.remainder.match(/^($|[\s,;\)\]\}])/)) { | ||
|  |           return a; | ||
|  |         } //  AND end of 'phrase'
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         var m = input.match(/^(?:\((?:\\ca\s?)?\$[amothc]\$\))/); // OR crystal system ($o$) (\ca$c$)
 | ||
|  | 
 | ||
|  |         if (m) { | ||
|  |           return { | ||
|  |             match_: m[0], | ||
|  |             remainder: input.substr(m[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }, | ||
|  |       '_{(state of aggregation)}$': /^_\{(\([a-z]{1,3}\))\}/, | ||
|  |       '{[(': /^(?:\\\{|\[|\()/, | ||
|  |       ')]}': /^(?:\)|\]|\\\})/, | ||
|  |       ', ': /^[,;]\s*/, | ||
|  |       ',': /^[,;]/, | ||
|  |       '.': /^[.]/, | ||
|  |       '. ': /^([.\u22C5\u00B7\u2022])\s*/, | ||
|  |       '...': /^\.\.\.(?=$|[^.])/, | ||
|  |       '* ': /^([*])\s*/, | ||
|  |       '^{(...)}': function _(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "^{", "", "", "}"); | ||
|  |       }, | ||
|  |       '^($...$)': function $$(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "^", "$", "$", ""); | ||
|  |       }, | ||
|  |       '^a': /^\^([0-9]+|[^\\_])/, | ||
|  |       '^\\x{}{}': function x(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "^", /^\\[a-zA-Z]+\{/, "}", "", "", "{", "}", "", true); | ||
|  |       }, | ||
|  |       '^\\x{}': function x(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "^", /^\\[a-zA-Z]+\{/, "}", ""); | ||
|  |       }, | ||
|  |       '^\\x': /^\^(\\[a-zA-Z]+)\s*/, | ||
|  |       '^(-1)': /^\^(-?\d+)/, | ||
|  |       '\'': /^'/, | ||
|  |       '_{(...)}': function _(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "_{", "", "", "}"); | ||
|  |       }, | ||
|  |       '_($...$)': function _$$(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "_", "$", "$", ""); | ||
|  |       }, | ||
|  |       '_9': /^_([+\-]?[0-9]+|[^\\])/, | ||
|  |       '_\\x{}{}': function _X(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "_", /^\\[a-zA-Z]+\{/, "}", "", "", "{", "}", "", true); | ||
|  |       }, | ||
|  |       '_\\x{}': function _X(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "_", /^\\[a-zA-Z]+\{/, "}", ""); | ||
|  |       }, | ||
|  |       '_\\x': /^_(\\[a-zA-Z]+)\s*/, | ||
|  |       '^_': /^(?:\^(?=_)|\_(?=\^)|[\^_]$)/, | ||
|  |       '{}': /^\{\}/, | ||
|  |       '{...}': function _(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "", "{", "}", ""); | ||
|  |       }, | ||
|  |       '{(...)}': function _(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '$...$': function $$(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "", "$", "$", ""); | ||
|  |       }, | ||
|  |       '${(...)}$': function $$(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "${", "", "", "}$"); | ||
|  |       }, | ||
|  |       '$(...)$': function $$(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "$", "", "", "$"); | ||
|  |       }, | ||
|  |       '=<>': /^[=<>]/, | ||
|  |       '#': /^[#\u2261]/, | ||
|  |       '+': /^\+/, | ||
|  |       '-$': /^-(?=[\s_},;\]/]|$|\([a-z]+\))/, | ||
|  |       // -space -, -; -] -/ -$ -state-of-aggregation
 | ||
|  |       '-9': /^-(?=[0-9])/, | ||
|  |       '- orbital overlap': /^-(?=(?:[spd]|sp)(?:$|[\s,;\)\]\}]))/, | ||
|  |       '-': /^-/, | ||
|  |       'pm-operator': /^(?:\\pm|\$\\pm\$|\+-|\+\/-)/, | ||
|  |       'operator': /^(?:\+|(?:[\-=<>]|<<|>>|\\approx|\$\\approx\$)(?=\s|$|-?[0-9]))/, | ||
|  |       'arrowUpDown': /^(?:v|\(v\)|\^|\(\^\))(?=$|[\s,;\)\]\}])/, | ||
|  |       '\\bond{(...)}': function bond(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\bond{", "", "", "}"); | ||
|  |       }, | ||
|  |       '->': /^(?:<->|<-->|->|<-|<=>>|<<=>|<=>|[\u2192\u27F6\u21CC])/, | ||
|  |       'CMT': /^[CMT](?=\[)/, | ||
|  |       '[(...)]': function _(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "[", "", "", "]"); | ||
|  |       }, | ||
|  |       '1st-level escape': /^(&|\\\\|\\hline)\s*/, | ||
|  |       '\\,': /^(?:\\[,\ ;:])/, | ||
|  |       // \\x - but output no space before
 | ||
|  |       '\\x{}{}': function x(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "", /^\\[a-zA-Z]+\{/, "}", "", "", "{", "}", "", true); | ||
|  |       }, | ||
|  |       '\\x{}': function x(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "", /^\\[a-zA-Z]+\{/, "}", ""); | ||
|  |       }, | ||
|  |       '\\ca': /^\\ca(?:\s+|(?![a-zA-Z]))/, | ||
|  |       '\\x': /^(?:\\[a-zA-Z]+\s*|\\[_&{}%])/, | ||
|  |       'orbital': /^(?:[0-9]{1,2}[spdfgh]|[0-9]{0,2}sp)(?=$|[^a-zA-Z])/, | ||
|  |       // only those with numbers in front, because the others will be formatted correctly anyway
 | ||
|  |       'others': /^[\/~|]/, | ||
|  |       '\\frac{(...)}': function frac(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\frac{", "", "", "}", "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\overset{(...)}': function overset(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\overset{", "", "", "}", "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\underset{(...)}': function underset(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\underset{", "", "", "}", "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\underbrace{(...)}': function underbrace(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\underbrace{", "", "", "}_", "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\color{(...)}0': function color0(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\color{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\color{(...)}{(...)}1': function color1(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\color{", "", "", "}", "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\color(...){(...)}2': function color2(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\color", "\\", "", /^(?=\{)/, "{", "", "", "}"); | ||
|  |       }, | ||
|  |       '\\ce{(...)}': function ce(input) { | ||
|  |         return mhchemParser.patterns.findObserveGroups(input, "\\ce{", "", "", "}"); | ||
|  |       }, | ||
|  |       'oxidation$': /^(?:[+-][IVX]+|\\pm\s*0|\$\\pm\$\s*0)$/, | ||
|  |       'd-oxidation$': /^(?:[+-]?\s?[IVX]+|\\pm\s*0|\$\\pm\$\s*0)$/, | ||
|  |       // 0 could be oxidation or charge
 | ||
|  |       'roman numeral': /^[IVX]+/, | ||
|  |       '1/2$': /^[+\-]?(?:[0-9]+|\$[a-z]\$|[a-z])\/[0-9]+(?:\$[a-z]\$|[a-z])?$/, | ||
|  |       'amount': function amount(input) { | ||
|  |         var match; // e.g. 2, 0.5, 1/2, -2, n/2, +;  $a$ could be added later in parsing
 | ||
|  | 
 | ||
|  |         match = input.match(/^(?:(?:(?:\([+\-]?[0-9]+\/[0-9]+\)|[+\-]?(?:[0-9]+|\$[a-z]\$|[a-z])\/[0-9]+|[+\-]?[0-9]+[.,][0-9]+|[+\-]?\.[0-9]+|[+\-]?[0-9]+)(?:[a-z](?=\s*[A-Z]))?)|[+\-]?[a-z](?=\s*[A-Z])|\+(?!\s))/); | ||
|  | 
 | ||
|  |         if (match) { | ||
|  |           return { | ||
|  |             match_: match[0], | ||
|  |             remainder: input.substr(match[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         var a = mhchemParser.patterns.findObserveGroups(input, "", "$", "$", ""); | ||
|  | 
 | ||
|  |         if (a) { | ||
|  |           // e.g. $2n-1$, $-$
 | ||
|  |           match = a.match_.match(/^\$(?:\(?[+\-]?(?:[0-9]*[a-z]?[+\-])?[0-9]*[a-z](?:[+\-][0-9]*[a-z]?)?\)?|\+|-)\$$/); | ||
|  | 
 | ||
|  |           if (match) { | ||
|  |             return { | ||
|  |               match_: match[0], | ||
|  |               remainder: input.substr(match[0].length) | ||
|  |             }; | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }, | ||
|  |       'amount2': function amount2(input) { | ||
|  |         return this['amount'](input); | ||
|  |       }, | ||
|  |       '(KV letters),': /^(?:[A-Z][a-z]{0,2}|i)(?=,)/, | ||
|  |       'formula$': function formula$(input) { | ||
|  |         if (input.match(/^\([a-z]+\)$/)) { | ||
|  |           return null; | ||
|  |         } // state of aggregation = no formula
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         var match = input.match(/^(?:[a-z]|(?:[0-9\ \+\-\,\.\(\)]+[a-z])+[0-9\ \+\-\,\.\(\)]*|(?:[a-z][0-9\ \+\-\,\.\(\)]+)+[a-z]?)$/); | ||
|  | 
 | ||
|  |         if (match) { | ||
|  |           return { | ||
|  |             match_: match[0], | ||
|  |             remainder: input.substr(match[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }, | ||
|  |       'uprightEntities': /^(?:pH|pOH|pC|pK|iPr|iBu)(?=$|[^a-zA-Z])/, | ||
|  |       '/': /^\s*(\/)\s*/, | ||
|  |       '//': /^\s*(\/\/)\s*/, | ||
|  |       '*': /^\s*[*.]\s*/ | ||
|  |     }, | ||
|  |     findObserveGroups: function findObserveGroups(input, begExcl, begIncl, endIncl, endExcl, beg2Excl, beg2Incl, end2Incl, end2Excl, combine) { | ||
|  |       /** @type {{(input: string, pattern: string | RegExp): string | string[] | null;}} */ | ||
|  |       var _match = function _match(input, pattern) { | ||
|  |         if (typeof pattern === "string") { | ||
|  |           if (input.indexOf(pattern) !== 0) { | ||
|  |             return null; | ||
|  |           } | ||
|  | 
 | ||
|  |           return pattern; | ||
|  |         } else { | ||
|  |           var match = input.match(pattern); | ||
|  | 
 | ||
|  |           if (!match) { | ||
|  |             return null; | ||
|  |           } | ||
|  | 
 | ||
|  |           return match[0]; | ||
|  |         } | ||
|  |       }; | ||
|  |       /** @type {{(input: string, i: number, endChars: string | RegExp): {endMatchBegin: number, endMatchEnd: number} | null;}} */ | ||
|  | 
 | ||
|  | 
 | ||
|  |       var _findObserveGroups = function _findObserveGroups(input, i, endChars) { | ||
|  |         var braces = 0; | ||
|  | 
 | ||
|  |         while (i < input.length) { | ||
|  |           var a = input.charAt(i); | ||
|  | 
 | ||
|  |           var match = _match(input.substr(i), endChars); | ||
|  | 
 | ||
|  |           if (match !== null && braces === 0) { | ||
|  |             return { | ||
|  |               endMatchBegin: i, | ||
|  |               endMatchEnd: i + match.length | ||
|  |             }; | ||
|  |           } else if (a === "{") { | ||
|  |             braces++; | ||
|  |           } else if (a === "}") { | ||
|  |             if (braces === 0) { | ||
|  |               throw ["ExtraCloseMissingOpen", "Extra close brace or missing open brace"]; | ||
|  |             } else { | ||
|  |               braces--; | ||
|  |             } | ||
|  |           } | ||
|  | 
 | ||
|  |           i++; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (braces > 0) { | ||
|  |           return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       }; | ||
|  | 
 | ||
|  |       var match = _match(input, begExcl); | ||
|  | 
 | ||
|  |       if (match === null) { | ||
|  |         return null; | ||
|  |       } | ||
|  | 
 | ||
|  |       input = input.substr(match.length); | ||
|  |       match = _match(input, begIncl); | ||
|  | 
 | ||
|  |       if (match === null) { | ||
|  |         return null; | ||
|  |       } | ||
|  | 
 | ||
|  |       var e = _findObserveGroups(input, match.length, endIncl || endExcl); | ||
|  | 
 | ||
|  |       if (e === null) { | ||
|  |         return null; | ||
|  |       } | ||
|  | 
 | ||
|  |       var match1 = input.substring(0, endIncl ? e.endMatchEnd : e.endMatchBegin); | ||
|  | 
 | ||
|  |       if (!(beg2Excl || beg2Incl)) { | ||
|  |         return { | ||
|  |           match_: match1, | ||
|  |           remainder: input.substr(e.endMatchEnd) | ||
|  |         }; | ||
|  |       } else { | ||
|  |         var group2 = this.findObserveGroups(input.substr(e.endMatchEnd), beg2Excl, beg2Incl, end2Incl, end2Excl); | ||
|  | 
 | ||
|  |         if (group2 === null) { | ||
|  |           return null; | ||
|  |         } | ||
|  |         /** @type {string[]} */ | ||
|  | 
 | ||
|  | 
 | ||
|  |         var matchRet = [match1, group2.match_]; | ||
|  |         return { | ||
|  |           match_: combine ? matchRet.join("") : matchRet, | ||
|  |           remainder: group2.remainder | ||
|  |         }; | ||
|  |       } | ||
|  |     }, | ||
|  |     //
 | ||
|  |     // Matching function
 | ||
|  |     // e.g. match("a", input) will look for the regexp called "a" and see if it matches
 | ||
|  |     // returns null or {match_:"a", remainder:"bc"}
 | ||
|  |     //
 | ||
|  |     match_: function match_(m, input) { | ||
|  |       var pattern = mhchemParser.patterns.patterns[m]; | ||
|  | 
 | ||
|  |       if (pattern === undefined) { | ||
|  |         throw ["MhchemBugP", "mhchem bug P. Please report. (" + m + ")"]; // Trying to use non-existing pattern
 | ||
|  |       } else if (typeof pattern === "function") { | ||
|  |         return mhchemParser.patterns.patterns[m](input); // cannot use cached var pattern here, because some pattern functions need this===mhchemParser
 | ||
|  |       } else { | ||
|  |         // RegExp
 | ||
|  |         var match = input.match(pattern); | ||
|  | 
 | ||
|  |         if (match) { | ||
|  |           var mm; | ||
|  | 
 | ||
|  |           if (match[2]) { | ||
|  |             mm = [match[1], match[2]]; | ||
|  |           } else if (match[1]) { | ||
|  |             mm = match[1]; | ||
|  |           } else { | ||
|  |             mm = match[0]; | ||
|  |           } | ||
|  | 
 | ||
|  |           return { | ||
|  |             match_: mm, | ||
|  |             remainder: input.substr(match[0].length) | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         return null; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   //
 | ||
|  |   // Generic state machine actions
 | ||
|  |   //
 | ||
|  |   actions: { | ||
|  |     'a=': function a(buffer, m) { | ||
|  |       buffer.a = (buffer.a || "") + m; | ||
|  |     }, | ||
|  |     'b=': function b(buffer, m) { | ||
|  |       buffer.b = (buffer.b || "") + m; | ||
|  |     }, | ||
|  |     'p=': function p(buffer, m) { | ||
|  |       buffer.p = (buffer.p || "") + m; | ||
|  |     }, | ||
|  |     'o=': function o(buffer, m) { | ||
|  |       buffer.o = (buffer.o || "") + m; | ||
|  |     }, | ||
|  |     'q=': function q(buffer, m) { | ||
|  |       buffer.q = (buffer.q || "") + m; | ||
|  |     }, | ||
|  |     'd=': function d(buffer, m) { | ||
|  |       buffer.d = (buffer.d || "") + m; | ||
|  |     }, | ||
|  |     'rm=': function rm(buffer, m) { | ||
|  |       buffer.rm = (buffer.rm || "") + m; | ||
|  |     }, | ||
|  |     'text=': function text(buffer, m) { | ||
|  |       buffer.text_ = (buffer.text_ || "") + m; | ||
|  |     }, | ||
|  |     'insert': function insert(buffer, m, a) { | ||
|  |       return { | ||
|  |         type_: a | ||
|  |       }; | ||
|  |     }, | ||
|  |     'insert+p1': function insertP1(buffer, m, a) { | ||
|  |       return { | ||
|  |         type_: a, | ||
|  |         p1: m | ||
|  |       }; | ||
|  |     }, | ||
|  |     'insert+p1+p2': function insertP1P2(buffer, m, a) { | ||
|  |       return { | ||
|  |         type_: a, | ||
|  |         p1: m[0], | ||
|  |         p2: m[1] | ||
|  |       }; | ||
|  |     }, | ||
|  |     'copy': function copy(buffer, m) { | ||
|  |       return m; | ||
|  |     }, | ||
|  |     'rm': function rm(buffer, m) { | ||
|  |       return { | ||
|  |         type_: 'rm', | ||
|  |         p1: m || "" | ||
|  |       }; | ||
|  |     }, | ||
|  |     'text': function text(buffer, m) { | ||
|  |       return mhchemParser.go(m, 'text'); | ||
|  |     }, | ||
|  |     '{text}': function text(buffer, m) { | ||
|  |       var ret = ["{"]; | ||
|  |       mhchemParser.concatArray(ret, mhchemParser.go(m, 'text')); | ||
|  |       ret.push("}"); | ||
|  |       return ret; | ||
|  |     }, | ||
|  |     'tex-math': function texMath(buffer, m) { | ||
|  |       return mhchemParser.go(m, 'tex-math'); | ||
|  |     }, | ||
|  |     'tex-math tight': function texMathTight(buffer, m) { | ||
|  |       return mhchemParser.go(m, 'tex-math tight'); | ||
|  |     }, | ||
|  |     'bond': function bond(buffer, m, k) { | ||
|  |       return { | ||
|  |         type_: 'bond', | ||
|  |         kind_: k || m | ||
|  |       }; | ||
|  |     }, | ||
|  |     'color0-output': function color0Output(buffer, m) { | ||
|  |       return { | ||
|  |         type_: 'color0', | ||
|  |         color: m[0] | ||
|  |       }; | ||
|  |     }, | ||
|  |     'ce': function ce(buffer, m) { | ||
|  |       return mhchemParser.go(m); | ||
|  |     }, | ||
|  |     '1/2': function _(buffer, m) { | ||
|  |       /** @type {ParserOutput[]} */ | ||
|  |       var ret = []; | ||
|  | 
 | ||
|  |       if (m.match(/^[+\-]/)) { | ||
|  |         ret.push(m.substr(0, 1)); | ||
|  |         m = m.substr(1); | ||
|  |       } | ||
|  | 
 | ||
|  |       var n = m.match(/^([0-9]+|\$[a-z]\$|[a-z])\/([0-9]+)(\$[a-z]\$|[a-z])?$/); | ||
|  |       n[1] = n[1].replace(/\$/g, ""); | ||
|  |       ret.push({ | ||
|  |         type_: 'frac', | ||
|  |         p1: n[1], | ||
|  |         p2: n[2] | ||
|  |       }); | ||
|  | 
 | ||
|  |       if (n[3]) { | ||
|  |         n[3] = n[3].replace(/\$/g, ""); | ||
|  |         ret.push({ | ||
|  |           type_: 'tex-math', | ||
|  |           p1: n[3] | ||
|  |         }); | ||
|  |       } | ||
|  | 
 | ||
|  |       return ret; | ||
|  |     }, | ||
|  |     '9,9': function _(buffer, m) { | ||
|  |       return mhchemParser.go(m, '9,9'); | ||
|  |     } | ||
|  |   }, | ||
|  |   //
 | ||
|  |   // createTransitions
 | ||
|  |   // convert  { 'letter': { 'state': { action_: 'output' } } }  to  { 'state' => [ { pattern: 'letter', task: { action_: [{type_: 'output'}] } } ] }
 | ||
|  |   // with expansion of 'a|b' to 'a' and 'b' (at 2 places)
 | ||
|  |   //
 | ||
|  |   createTransitions: function createTransitions(o) { | ||
|  |     var pattern, state; | ||
|  |     /** @type {string[]} */ | ||
|  | 
 | ||
|  |     var stateArray; | ||
|  |     var i; //
 | ||
|  |     // 1. Collect all states
 | ||
|  |     //
 | ||
|  | 
 | ||
|  |     /** @type {Transitions} */ | ||
|  | 
 | ||
|  |     var transitions = {}; | ||
|  | 
 | ||
|  |     for (pattern in o) { | ||
|  |       for (state in o[pattern]) { | ||
|  |         stateArray = state.split("|"); | ||
|  |         o[pattern][state].stateArray = stateArray; | ||
|  | 
 | ||
|  |         for (i = 0; i < stateArray.length; i++) { | ||
|  |           transitions[stateArray[i]] = []; | ||
|  |         } | ||
|  |       } | ||
|  |     } //
 | ||
|  |     // 2. Fill states
 | ||
|  |     //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |     for (pattern in o) { | ||
|  |       for (state in o[pattern]) { | ||
|  |         stateArray = o[pattern][state].stateArray || []; | ||
|  | 
 | ||
|  |         for (i = 0; i < stateArray.length; i++) { | ||
|  |           //
 | ||
|  |           // 2a. Normalize actions into array:  'text=' ==> [{type_:'text='}]
 | ||
|  |           // (Note to myself: Resolving the function here would be problematic. It would need .bind (for *this*) and currying (for *option*).)
 | ||
|  |           //
 | ||
|  | 
 | ||
|  |           /** @type {any} */ | ||
|  |           var p = o[pattern][state]; | ||
|  | 
 | ||
|  |           if (p.action_) { | ||
|  |             p.action_ = [].concat(p.action_); | ||
|  | 
 | ||
|  |             for (var k = 0; k < p.action_.length; k++) { | ||
|  |               if (typeof p.action_[k] === "string") { | ||
|  |                 p.action_[k] = { | ||
|  |                   type_: p.action_[k] | ||
|  |                 }; | ||
|  |               } | ||
|  |             } | ||
|  |           } else { | ||
|  |             p.action_ = []; | ||
|  |           } //
 | ||
|  |           // 2.b Multi-insert
 | ||
|  |           //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |           var patternArray = pattern.split("|"); | ||
|  | 
 | ||
|  |           for (var j = 0; j < patternArray.length; j++) { | ||
|  |             if (stateArray[i] === '*') { | ||
|  |               // insert into all
 | ||
|  |               for (var t in transitions) { | ||
|  |                 transitions[t].push({ | ||
|  |                   pattern: patternArray[j], | ||
|  |                   task: p | ||
|  |                 }); | ||
|  |               } | ||
|  |             } else { | ||
|  |               transitions[stateArray[i]].push({ | ||
|  |                 pattern: patternArray[j], | ||
|  |                 task: p | ||
|  |               }); | ||
|  |             } | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     return transitions; | ||
|  |   }, | ||
|  |   stateMachines: {} | ||
|  | }; //
 | ||
|  | // Definition of state machines
 | ||
|  | //
 | ||
|  | 
 | ||
|  | mhchemParser.stateMachines = { | ||
|  |   //
 | ||
|  |   // \ce state machines
 | ||
|  |   //
 | ||
|  |   //#region ce
 | ||
|  |   'ce': { | ||
|  |     // main parser
 | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '0|1|2': { | ||
|  |           action_: 'beginsWithBond=false', | ||
|  |           revisit: true, | ||
|  |           toContinue: true | ||
|  |         } | ||
|  |       }, | ||
|  |       'oxidation$': { | ||
|  |         '0': { | ||
|  |           action_: 'oxidation-output' | ||
|  |         } | ||
|  |       }, | ||
|  |       'CMT': { | ||
|  |         'r': { | ||
|  |           action_: 'rdt=', | ||
|  |           nextState: 'rt' | ||
|  |         }, | ||
|  |         'rd': { | ||
|  |           action_: 'rqt=', | ||
|  |           nextState: 'rdt' | ||
|  |         } | ||
|  |       }, | ||
|  |       'arrowUpDown': { | ||
|  |         '0|1|2|as': { | ||
|  |           action_: ['sb=false', 'output', 'operator'], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       'uprightEntities': { | ||
|  |         '0|1|2': { | ||
|  |           action_: ['o=', 'output'], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       'orbital': { | ||
|  |         '0|1|2|3': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       '->': { | ||
|  |         '0|1|2|3': { | ||
|  |           action_: 'r=', | ||
|  |           nextState: 'r' | ||
|  |         }, | ||
|  |         'a|as': { | ||
|  |           action_: ['output', 'r='], | ||
|  |           nextState: 'r' | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', 'r='], | ||
|  |           nextState: 'r' | ||
|  |         } | ||
|  |       }, | ||
|  |       '+': { | ||
|  |         'o': { | ||
|  |           action_: 'd= kv', | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         'd|D': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'qd|qD': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'dq': { | ||
|  |           action_: ['output', 'd='], | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         '3': { | ||
|  |           action_: ['sb=false', 'output', 'operator'], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       'amount': { | ||
|  |         '0|2': { | ||
|  |           action_: 'a=', | ||
|  |           nextState: 'a' | ||
|  |         } | ||
|  |       }, | ||
|  |       'pm-operator': { | ||
|  |         '0|1|2|a|as': { | ||
|  |           action_: ['sb=false', 'output', { | ||
|  |             type_: 'operator', | ||
|  |             option: '\\pm' | ||
|  |           }], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       'operator': { | ||
|  |         '0|1|2|a|as': { | ||
|  |           action_: ['sb=false', 'output', 'operator'], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-$': { | ||
|  |         'o|q': { | ||
|  |           action_: ['charge or bond', 'output'], | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'd': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         'D': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'qd': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'qD|dq': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-9': { | ||
|  |         '3|o': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert', | ||
|  |             option: 'hyphen' | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '- orbital overlap': { | ||
|  |         'o': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert', | ||
|  |             option: 'hyphen' | ||
|  |           }], | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         'd': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert', | ||
|  |             option: 'hyphen' | ||
|  |           }], | ||
|  |           nextState: '2' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-': { | ||
|  |         '0|1|2': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, 'beginsWithBond=true', { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         }, | ||
|  |         '3': { | ||
|  |           action_: { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           } | ||
|  |         }, | ||
|  |         'a': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert', | ||
|  |             option: 'hyphen' | ||
|  |           }], | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         'as': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         }, | ||
|  |         'b': { | ||
|  |           action_: 'b=' | ||
|  |         }, | ||
|  |         'o': { | ||
|  |           action_: { | ||
|  |             type_: '- after o/d', | ||
|  |             option: false | ||
|  |           }, | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: { | ||
|  |             type_: '- after o/d', | ||
|  |             option: false | ||
|  |           }, | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         'd|qd|dq': { | ||
|  |           action_: { | ||
|  |             type_: '- after o/d', | ||
|  |             option: true | ||
|  |           }, | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         'D|qD|p': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'bond', | ||
|  |             option: "-" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       'amount2': { | ||
|  |         '1|3': { | ||
|  |           action_: 'a=', | ||
|  |           nextState: 'a' | ||
|  |         } | ||
|  |       }, | ||
|  |       'letters': { | ||
|  |         '0|1|2|3|a|as|b|p|bp|o': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         'q|dq': { | ||
|  |           action_: ['output', 'o='], | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         'd|D|qd|qD': { | ||
|  |           action_: 'o after d', | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       'digits': { | ||
|  |         'o': { | ||
|  |           action_: 'q=', | ||
|  |           nextState: 'q' | ||
|  |         }, | ||
|  |         'd|D': { | ||
|  |           action_: 'q=', | ||
|  |           nextState: 'dq' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: ['output', 'o='], | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         'a': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       'space A': { | ||
|  |         'b|p|bp': {} | ||
|  |       }, | ||
|  |       'space': { | ||
|  |         'a': { | ||
|  |           nextState: 'as' | ||
|  |         }, | ||
|  |         '0': { | ||
|  |           action_: 'sb=false' | ||
|  |         }, | ||
|  |         '1|2': { | ||
|  |           action_: 'sb=true' | ||
|  |         }, | ||
|  |         'r|rt|rd|rdt|rdq': { | ||
|  |           action_: 'output', | ||
|  |           nextState: '0' | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', 'sb=true'], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       '1st-level escape': { | ||
|  |         '1|2': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert+p1', | ||
|  |             option: '1st-level escape' | ||
|  |           }] | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert+p1', | ||
|  |             option: '1st-level escape' | ||
|  |           }], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '[(...)]': { | ||
|  |         'r|rt': { | ||
|  |           action_: 'rd=', | ||
|  |           nextState: 'rd' | ||
|  |         }, | ||
|  |         'rd|rdt': { | ||
|  |           action_: 'rq=', | ||
|  |           nextState: 'rdq' | ||
|  |         } | ||
|  |       }, | ||
|  |       '...': { | ||
|  |         'o|d|D|dq|qd|qD': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'bond', | ||
|  |             option: "..." | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, { | ||
|  |             type_: 'insert', | ||
|  |             option: 'ellipsis' | ||
|  |           }], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       '. |* ': { | ||
|  |         '*': { | ||
|  |           action_: ['output', { | ||
|  |             type_: 'insert', | ||
|  |             option: 'addition compound' | ||
|  |           }], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       'state of aggregation $': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'state of aggregation'], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{[(': { | ||
|  |         'a|as|o': { | ||
|  |           action_: ['o=', 'output', 'parenthesisLevel++'], | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         '0|1|2|3': { | ||
|  |           action_: ['o=', 'output', 'parenthesisLevel++'], | ||
|  |           nextState: '2' | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', 'o=', 'output', 'parenthesisLevel++'], | ||
|  |           nextState: '2' | ||
|  |         } | ||
|  |       }, | ||
|  |       ')]}': { | ||
|  |         '0|1|2|3|b|p|bp|o': { | ||
|  |           action_: ['o=', 'parenthesisLevel--'], | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         'a|as|d|D|q|qd|qD|dq': { | ||
|  |           action_: ['output', 'o=', 'parenthesisLevel--'], | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       ', ': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'comma'], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '^_': { | ||
|  |         // ^ and _ without a sensible argument
 | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       '^{(...)}|^($...$)': { | ||
|  |         '0|1|2|as': { | ||
|  |           action_: 'b=', | ||
|  |           nextState: 'b' | ||
|  |         }, | ||
|  |         'p': { | ||
|  |           action_: 'b=', | ||
|  |           nextState: 'bp' | ||
|  |         }, | ||
|  |         '3|o': { | ||
|  |           action_: 'd= kv', | ||
|  |           nextState: 'D' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qD' | ||
|  |         }, | ||
|  |         'd|D|qd|qD|dq': { | ||
|  |           action_: ['output', 'd='], | ||
|  |           nextState: 'D' | ||
|  |         } | ||
|  |       }, | ||
|  |       '^a|^\\x{}{}|^\\x{}|^\\x|\'': { | ||
|  |         '0|1|2|as': { | ||
|  |           action_: 'b=', | ||
|  |           nextState: 'b' | ||
|  |         }, | ||
|  |         'p': { | ||
|  |           action_: 'b=', | ||
|  |           nextState: 'bp' | ||
|  |         }, | ||
|  |         '3|o': { | ||
|  |           action_: 'd= kv', | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         'q': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'qd' | ||
|  |         }, | ||
|  |         'd|qd|D|qD': { | ||
|  |           action_: 'd=' | ||
|  |         }, | ||
|  |         'dq': { | ||
|  |           action_: ['output', 'd='], | ||
|  |           nextState: 'd' | ||
|  |         } | ||
|  |       }, | ||
|  |       '_{(state of aggregation)}$': { | ||
|  |         'd|D|q|qd|qD|dq': { | ||
|  |           action_: ['output', 'q='], | ||
|  |           nextState: 'q' | ||
|  |         } | ||
|  |       }, | ||
|  |       '_{(...)}|_($...$)|_9|_\\x{}{}|_\\x{}|_\\x': { | ||
|  |         '0|1|2|as': { | ||
|  |           action_: 'p=', | ||
|  |           nextState: 'p' | ||
|  |         }, | ||
|  |         'b': { | ||
|  |           action_: 'p=', | ||
|  |           nextState: 'bp' | ||
|  |         }, | ||
|  |         '3|o': { | ||
|  |           action_: 'q=', | ||
|  |           nextState: 'q' | ||
|  |         }, | ||
|  |         'd|D': { | ||
|  |           action_: 'q=', | ||
|  |           nextState: 'dq' | ||
|  |         }, | ||
|  |         'q|qd|qD|dq': { | ||
|  |           action_: ['output', 'q='], | ||
|  |           nextState: 'q' | ||
|  |         } | ||
|  |       }, | ||
|  |       '=<>': { | ||
|  |         '0|1|2|3|a|as|o|q|d|D|qd|qD|dq': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'bond'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '#': { | ||
|  |         '0|1|2|3|a|as|o': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, { | ||
|  |             type_: 'bond', | ||
|  |             option: "#" | ||
|  |           }], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{}': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}': { | ||
|  |         '0|1|2|3|a|as|b|p|bp': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         'o|d|D|q|qd|qD|dq': { | ||
|  |           action_: ['output', 'o='], | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       '$...$': { | ||
|  |         'a': { | ||
|  |           action_: 'a=' | ||
|  |         }, | ||
|  |         // 2$n$
 | ||
|  |         '0|1|2|3|as|b|p|bp|o': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: 'o' | ||
|  |         }, | ||
|  |         // not 'amount'
 | ||
|  |         'as|o': { | ||
|  |           action_: 'o=' | ||
|  |         }, | ||
|  |         'q|d|D|qd|qD|dq': { | ||
|  |           action_: ['output', 'o='], | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\bond{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'bond'], | ||
|  |           nextState: "3" | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\frac{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, 'frac-output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\overset{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'overset-output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\underset{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'underset-output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\underbrace{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'underbrace-output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}{(...)}1|\\color(...){(...)}2': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'color-output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}0': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'color0-output'] | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ce{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 2 | ||
|  |           }, 'ce'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\,': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, 'copy'], | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\x{}{}|\\x{}|\\x': { | ||
|  |         '0|1|2|3|a|as|b|p|bp|o|c0': { | ||
|  |           action_: ['o=', 'output'], | ||
|  |           nextState: '3' | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', 'o=', 'output'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       'others': { | ||
|  |         '*': { | ||
|  |           action_: [{ | ||
|  |             type_: 'output', | ||
|  |             option: 1 | ||
|  |           }, 'copy'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else2': { | ||
|  |         'a': { | ||
|  |           action_: 'a to o', | ||
|  |           nextState: 'o', | ||
|  |           revisit: true | ||
|  |         }, | ||
|  |         'as': { | ||
|  |           action_: ['output', 'sb=true'], | ||
|  |           nextState: '1', | ||
|  |           revisit: true | ||
|  |         }, | ||
|  |         'r|rt|rd|rdt|rdq': { | ||
|  |           action_: ['output'], | ||
|  |           nextState: '0', | ||
|  |           revisit: true | ||
|  |         }, | ||
|  |         '*': { | ||
|  |           action_: ['output', 'copy'], | ||
|  |           nextState: '3' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'o after d': function oAfterD(buffer, m) { | ||
|  |         var ret; | ||
|  | 
 | ||
|  |         if ((buffer.d || "").match(/^[0-9]+$/)) { | ||
|  |           var tmp = buffer.d; | ||
|  |           buffer.d = undefined; | ||
|  |           ret = this['output'](buffer); | ||
|  |           buffer.b = tmp; | ||
|  |         } else { | ||
|  |           ret = this['output'](buffer); | ||
|  |         } | ||
|  | 
 | ||
|  |         mhchemParser.actions['o='](buffer, m); | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'd= kv': function dKv(buffer, m) { | ||
|  |         buffer.d = m; | ||
|  |         buffer.dType = 'kv'; | ||
|  |       }, | ||
|  |       'charge or bond': function chargeOrBond(buffer, m) { | ||
|  |         if (buffer['beginsWithBond']) { | ||
|  |           /** @type {ParserOutput[]} */ | ||
|  |           var ret = []; | ||
|  |           mhchemParser.concatArray(ret, this['output'](buffer)); | ||
|  |           mhchemParser.concatArray(ret, mhchemParser.actions['bond'](buffer, m, "-")); | ||
|  |           return ret; | ||
|  |         } else { | ||
|  |           buffer.d = m; | ||
|  |         } | ||
|  |       }, | ||
|  |       '- after o/d': function afterOD(buffer, m, isAfterD) { | ||
|  |         var c1 = mhchemParser.patterns.match_('orbital', buffer.o || ""); | ||
|  |         var c2 = mhchemParser.patterns.match_('one lowercase greek letter $', buffer.o || ""); | ||
|  |         var c3 = mhchemParser.patterns.match_('one lowercase latin letter $', buffer.o || ""); | ||
|  |         var c4 = mhchemParser.patterns.match_('$one lowercase latin letter$ $', buffer.o || ""); | ||
|  |         var hyphenFollows = m === "-" && (c1 && c1.remainder === "" || c2 || c3 || c4); | ||
|  | 
 | ||
|  |         if (hyphenFollows && !buffer.a && !buffer.b && !buffer.p && !buffer.d && !buffer.q && !c1 && c3) { | ||
|  |           buffer.o = '$' + buffer.o + '$'; | ||
|  |         } | ||
|  |         /** @type {ParserOutput[]} */ | ||
|  | 
 | ||
|  | 
 | ||
|  |         var ret = []; | ||
|  | 
 | ||
|  |         if (hyphenFollows) { | ||
|  |           mhchemParser.concatArray(ret, this['output'](buffer)); | ||
|  |           ret.push({ | ||
|  |             type_: 'hyphen' | ||
|  |           }); | ||
|  |         } else { | ||
|  |           c1 = mhchemParser.patterns.match_('digits', buffer.d || ""); | ||
|  | 
 | ||
|  |           if (isAfterD && c1 && c1.remainder === '') { | ||
|  |             mhchemParser.concatArray(ret, mhchemParser.actions['d='](buffer, m)); | ||
|  |             mhchemParser.concatArray(ret, this['output'](buffer)); | ||
|  |           } else { | ||
|  |             mhchemParser.concatArray(ret, this['output'](buffer)); | ||
|  |             mhchemParser.concatArray(ret, mhchemParser.actions['bond'](buffer, m, "-")); | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'a to o': function aToO(buffer) { | ||
|  |         buffer.o = buffer.a; | ||
|  |         buffer.a = undefined; | ||
|  |       }, | ||
|  |       'sb=true': function sbTrue(buffer) { | ||
|  |         buffer.sb = true; | ||
|  |       }, | ||
|  |       'sb=false': function sbFalse(buffer) { | ||
|  |         buffer.sb = false; | ||
|  |       }, | ||
|  |       'beginsWithBond=true': function beginsWithBondTrue(buffer) { | ||
|  |         buffer['beginsWithBond'] = true; | ||
|  |       }, | ||
|  |       'beginsWithBond=false': function beginsWithBondFalse(buffer) { | ||
|  |         buffer['beginsWithBond'] = false; | ||
|  |       }, | ||
|  |       'parenthesisLevel++': function parenthesisLevel(buffer) { | ||
|  |         buffer['parenthesisLevel']++; | ||
|  |       }, | ||
|  |       'parenthesisLevel--': function parenthesisLevel(buffer) { | ||
|  |         buffer['parenthesisLevel']--; | ||
|  |       }, | ||
|  |       'state of aggregation': function stateOfAggregation(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'state of aggregation', | ||
|  |           p1: mhchemParser.go(m, 'o') | ||
|  |         }; | ||
|  |       }, | ||
|  |       'comma': function comma(buffer, m) { | ||
|  |         var a = m.replace(/\s*$/, ''); | ||
|  |         var withSpace = a !== m; | ||
|  | 
 | ||
|  |         if (withSpace && buffer['parenthesisLevel'] === 0) { | ||
|  |           return { | ||
|  |             type_: 'comma enumeration L', | ||
|  |             p1: a | ||
|  |           }; | ||
|  |         } else { | ||
|  |           return { | ||
|  |             type_: 'comma enumeration M', | ||
|  |             p1: a | ||
|  |           }; | ||
|  |         } | ||
|  |       }, | ||
|  |       'output': function output(buffer, m, entityFollows) { | ||
|  |         // entityFollows:
 | ||
|  |         //   undefined = if we have nothing else to output, also ignore the just read space (buffer.sb)
 | ||
|  |         //   1 = an entity follows, never omit the space if there was one just read before (can only apply to state 1)
 | ||
|  |         //   2 = 1 + the entity can have an amount, so output a\, instead of converting it to o (can only apply to states a|as)
 | ||
|  | 
 | ||
|  |         /** @type {ParserOutput | ParserOutput[]} */ | ||
|  |         var ret; | ||
|  | 
 | ||
|  |         if (!buffer.r) { | ||
|  |           ret = []; | ||
|  | 
 | ||
|  |           if (!buffer.a && !buffer.b && !buffer.p && !buffer.o && !buffer.q && !buffer.d && !entityFollows) ; else { | ||
|  |             if (buffer.sb) { | ||
|  |               ret.push({ | ||
|  |                 type_: 'entitySkip' | ||
|  |               }); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (!buffer.o && !buffer.q && !buffer.d && !buffer.b && !buffer.p && entityFollows !== 2) { | ||
|  |               buffer.o = buffer.a; | ||
|  |               buffer.a = undefined; | ||
|  |             } else if (!buffer.o && !buffer.q && !buffer.d && (buffer.b || buffer.p)) { | ||
|  |               buffer.o = buffer.a; | ||
|  |               buffer.d = buffer.b; | ||
|  |               buffer.q = buffer.p; | ||
|  |               buffer.a = buffer.b = buffer.p = undefined; | ||
|  |             } else { | ||
|  |               if (buffer.o && buffer.dType === 'kv' && mhchemParser.patterns.match_('d-oxidation$', buffer.d || "")) { | ||
|  |                 buffer.dType = 'oxidation'; | ||
|  |               } else if (buffer.o && buffer.dType === 'kv' && !buffer.q) { | ||
|  |                 buffer.dType = undefined; | ||
|  |               } | ||
|  |             } | ||
|  | 
 | ||
|  |             ret.push({ | ||
|  |               type_: 'chemfive', | ||
|  |               a: mhchemParser.go(buffer.a, 'a'), | ||
|  |               b: mhchemParser.go(buffer.b, 'bd'), | ||
|  |               p: mhchemParser.go(buffer.p, 'pq'), | ||
|  |               o: mhchemParser.go(buffer.o, 'o'), | ||
|  |               q: mhchemParser.go(buffer.q, 'pq'), | ||
|  |               d: mhchemParser.go(buffer.d, buffer.dType === 'oxidation' ? 'oxidation' : 'bd'), | ||
|  |               dType: buffer.dType | ||
|  |             }); | ||
|  |           } | ||
|  |         } else { | ||
|  |           // r
 | ||
|  | 
 | ||
|  |           /** @type {ParserOutput[]} */ | ||
|  |           var rd; | ||
|  | 
 | ||
|  |           if (buffer.rdt === 'M') { | ||
|  |             rd = mhchemParser.go(buffer.rd, 'tex-math'); | ||
|  |           } else if (buffer.rdt === 'T') { | ||
|  |             rd = [{ | ||
|  |               type_: 'text', | ||
|  |               p1: buffer.rd || "" | ||
|  |             }]; | ||
|  |           } else { | ||
|  |             rd = mhchemParser.go(buffer.rd); | ||
|  |           } | ||
|  |           /** @type {ParserOutput[]} */ | ||
|  | 
 | ||
|  | 
 | ||
|  |           var rq; | ||
|  | 
 | ||
|  |           if (buffer.rqt === 'M') { | ||
|  |             rq = mhchemParser.go(buffer.rq, 'tex-math'); | ||
|  |           } else if (buffer.rqt === 'T') { | ||
|  |             rq = [{ | ||
|  |               type_: 'text', | ||
|  |               p1: buffer.rq || "" | ||
|  |             }]; | ||
|  |           } else { | ||
|  |             rq = mhchemParser.go(buffer.rq); | ||
|  |           } | ||
|  | 
 | ||
|  |           ret = { | ||
|  |             type_: 'arrow', | ||
|  |             r: buffer.r, | ||
|  |             rd: rd, | ||
|  |             rq: rq | ||
|  |           }; | ||
|  |         } | ||
|  | 
 | ||
|  |         for (var p in buffer) { | ||
|  |           if (p !== 'parenthesisLevel' && p !== 'beginsWithBond') { | ||
|  |             delete buffer[p]; | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'oxidation-output': function oxidationOutput(buffer, m) { | ||
|  |         var ret = ["{"]; | ||
|  |         mhchemParser.concatArray(ret, mhchemParser.go(m, 'oxidation')); | ||
|  |         ret.push("}"); | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'frac-output': function fracOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'frac-ce', | ||
|  |           p1: mhchemParser.go(m[0]), | ||
|  |           p2: mhchemParser.go(m[1]) | ||
|  |         }; | ||
|  |       }, | ||
|  |       'overset-output': function oversetOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'overset', | ||
|  |           p1: mhchemParser.go(m[0]), | ||
|  |           p2: mhchemParser.go(m[1]) | ||
|  |         }; | ||
|  |       }, | ||
|  |       'underset-output': function undersetOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'underset', | ||
|  |           p1: mhchemParser.go(m[0]), | ||
|  |           p2: mhchemParser.go(m[1]) | ||
|  |         }; | ||
|  |       }, | ||
|  |       'underbrace-output': function underbraceOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'underbrace', | ||
|  |           p1: mhchemParser.go(m[0]), | ||
|  |           p2: mhchemParser.go(m[1]) | ||
|  |         }; | ||
|  |       }, | ||
|  |       'color-output': function colorOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'color', | ||
|  |           color1: m[0], | ||
|  |           color2: mhchemParser.go(m[1]) | ||
|  |         }; | ||
|  |       }, | ||
|  |       'r=': function r(buffer, m) { | ||
|  |         buffer.r = m; | ||
|  |       }, | ||
|  |       'rdt=': function rdt(buffer, m) { | ||
|  |         buffer.rdt = m; | ||
|  |       }, | ||
|  |       'rd=': function rd(buffer, m) { | ||
|  |         buffer.rd = m; | ||
|  |       }, | ||
|  |       'rqt=': function rqt(buffer, m) { | ||
|  |         buffer.rqt = m; | ||
|  |       }, | ||
|  |       'rq=': function rq(buffer, m) { | ||
|  |         buffer.rq = m; | ||
|  |       }, | ||
|  |       'operator': function operator(buffer, m, p1) { | ||
|  |         return { | ||
|  |           type_: 'operator', | ||
|  |           kind_: p1 || m | ||
|  |         }; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'a': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       '1/2$': { | ||
|  |         '0': { | ||
|  |           action_: '1/2' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '0': { | ||
|  |           nextState: '1', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       '$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math tight', | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       }, | ||
|  |       ',': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert', | ||
|  |             option: 'commaDecimal' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       'else2': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: {} | ||
|  |   }, | ||
|  |   'o': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       '1/2$': { | ||
|  |         '0': { | ||
|  |           action_: '1/2' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '0': { | ||
|  |           nextState: '1', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       'letters': { | ||
|  |         '*': { | ||
|  |           action_: 'rm' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ca': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert', | ||
|  |             option: 'circa' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       }, | ||
|  |       '${(...)}$|$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: '{text}' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else2': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: {} | ||
|  |   }, | ||
|  |   'text': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}': { | ||
|  |         '*': { | ||
|  |           action_: 'text=' | ||
|  |         } | ||
|  |       }, | ||
|  |       '${(...)}$|$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\greek': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'rm'] | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\,|\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'copy'] | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'text=' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'output': function output(buffer) { | ||
|  |         if (buffer.text_) { | ||
|  |           /** @type {ParserOutput} */ | ||
|  |           var ret = { | ||
|  |             type_: 'text', | ||
|  |             p1: buffer.text_ | ||
|  |           }; | ||
|  | 
 | ||
|  |           for (var p in buffer) { | ||
|  |             delete buffer[p]; | ||
|  |           } | ||
|  | 
 | ||
|  |           return ret; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'pq': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       'state of aggregation $': { | ||
|  |         '*': { | ||
|  |           action_: 'state of aggregation' | ||
|  |         } | ||
|  |       }, | ||
|  |       'i$': { | ||
|  |         '0': { | ||
|  |           nextState: '!f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       '(KV letters),': { | ||
|  |         '0': { | ||
|  |           action_: 'rm', | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       'formula$': { | ||
|  |         '0': { | ||
|  |           nextState: 'f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       '1/2$': { | ||
|  |         '0': { | ||
|  |           action_: '1/2' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '0': { | ||
|  |           nextState: '!f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       '${(...)}$|$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: 'text' | ||
|  |         } | ||
|  |       }, | ||
|  |       'a-z': { | ||
|  |         'f': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       'letters': { | ||
|  |         '*': { | ||
|  |           action_: 'rm' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-9.,9': { | ||
|  |         '*': { | ||
|  |           action_: '9,9' | ||
|  |         } | ||
|  |       }, | ||
|  |       ',': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert+p1', | ||
|  |             option: 'comma enumeration S' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}{(...)}1|\\color(...){(...)}2': { | ||
|  |         '*': { | ||
|  |           action_: 'color-output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}0': { | ||
|  |         '*': { | ||
|  |           action_: 'color0-output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ce{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: 'ce' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\,|\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else2': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'state of aggregation': function stateOfAggregation(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'state of aggregation subscript', | ||
|  |           p1: mhchemParser.go(m, 'o') | ||
|  |         }; | ||
|  |       }, | ||
|  |       'color-output': function colorOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'color', | ||
|  |           color1: m[0], | ||
|  |           color2: mhchemParser.go(m[1], 'pq') | ||
|  |         }; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'bd': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       'x$': { | ||
|  |         '0': { | ||
|  |           nextState: '!f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       'formula$': { | ||
|  |         '0': { | ||
|  |           nextState: 'f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '0': { | ||
|  |           nextState: '!f', | ||
|  |           revisit: true | ||
|  |         } | ||
|  |       }, | ||
|  |       '-9.,9 no missing 0': { | ||
|  |         '*': { | ||
|  |           action_: '9,9' | ||
|  |         } | ||
|  |       }, | ||
|  |       '.': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert', | ||
|  |             option: 'electron dot' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       'a-z': { | ||
|  |         'f': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       'x': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert', | ||
|  |             option: 'KV x' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       'letters': { | ||
|  |         '*': { | ||
|  |           action_: 'rm' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\'': { | ||
|  |         '*': { | ||
|  |           action_: { | ||
|  |             type_: 'insert', | ||
|  |             option: 'prime' | ||
|  |           } | ||
|  |         } | ||
|  |       }, | ||
|  |       '${(...)}$|$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: 'text' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}{(...)}1|\\color(...){(...)}2': { | ||
|  |         '*': { | ||
|  |           action_: 'color-output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\color{(...)}0': { | ||
|  |         '*': { | ||
|  |           action_: 'color0-output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ce{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: 'ce' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\,|\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else2': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'color-output': function colorOutput(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'color', | ||
|  |           color1: m[0], | ||
|  |           color2: mhchemParser.go(m[1], 'bd') | ||
|  |         }; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'oxidation': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       'roman numeral': { | ||
|  |         '*': { | ||
|  |           action_: 'roman-numeral' | ||
|  |         } | ||
|  |       }, | ||
|  |       '${(...)}$|$(...)$': { | ||
|  |         '*': { | ||
|  |           action_: 'tex-math' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'roman-numeral': function romanNumeral(buffer, m) { | ||
|  |         return { | ||
|  |           type_: 'roman numeral', | ||
|  |           p1: m || "" | ||
|  |         }; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'tex-math': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ce{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'ce'] | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}|\\,|\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'o=' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'o=' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'output': function output(buffer) { | ||
|  |         if (buffer.o) { | ||
|  |           /** @type {ParserOutput} */ | ||
|  |           var ret = { | ||
|  |             type_: 'tex-math', | ||
|  |             p1: buffer.o | ||
|  |           }; | ||
|  | 
 | ||
|  |           for (var p in buffer) { | ||
|  |             delete buffer[p]; | ||
|  |           } | ||
|  | 
 | ||
|  |           return ret; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'tex-math tight': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\ce{(...)}': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'ce'] | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}|\\,|\\x{}{}|\\x{}|\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'o=' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-|+': { | ||
|  |         '*': { | ||
|  |           action_: 'tight operator' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'o=' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'tight operator': function tightOperator(buffer, m) { | ||
|  |         buffer.o = (buffer.o || "") + "{" + m + "}"; | ||
|  |       }, | ||
|  |       'output': function output(buffer) { | ||
|  |         if (buffer.o) { | ||
|  |           /** @type {ParserOutput} */ | ||
|  |           var ret = { | ||
|  |             type_: 'tex-math', | ||
|  |             p1: buffer.o | ||
|  |           }; | ||
|  | 
 | ||
|  |           for (var p in buffer) { | ||
|  |             delete buffer[p]; | ||
|  |           } | ||
|  | 
 | ||
|  |           return ret; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   '9,9': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': {} | ||
|  |       }, | ||
|  |       ',': { | ||
|  |         '*': { | ||
|  |           action_: 'comma' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'comma': function comma() { | ||
|  |         return { | ||
|  |           type_: 'commaDecimal' | ||
|  |         }; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   //#endregion
 | ||
|  |   //
 | ||
|  |   // \pu state machines
 | ||
|  |   //
 | ||
|  |   //#region pu
 | ||
|  |   'pu': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       'space$': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'space'] | ||
|  |         } | ||
|  |       }, | ||
|  |       '{[(|)]}': { | ||
|  |         '0|a': { | ||
|  |           action_: 'copy' | ||
|  |         } | ||
|  |       }, | ||
|  |       '(-)(9)^(-9)': { | ||
|  |         '0': { | ||
|  |           action_: 'number^', | ||
|  |           nextState: 'a' | ||
|  |         } | ||
|  |       }, | ||
|  |       '(-)(9.,9)(e)(99)': { | ||
|  |         '0': { | ||
|  |           action_: 'enumber', | ||
|  |           nextState: 'a' | ||
|  |         } | ||
|  |       }, | ||
|  |       'space': { | ||
|  |         '0|a': {} | ||
|  |       }, | ||
|  |       'pm-operator': { | ||
|  |         '0|a': { | ||
|  |           action_: { | ||
|  |             type_: 'operator', | ||
|  |             option: '\\pm' | ||
|  |           }, | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       'operator': { | ||
|  |         '0|a': { | ||
|  |           action_: 'copy', | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '//': { | ||
|  |         'd': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: '/' | ||
|  |         } | ||
|  |       }, | ||
|  |       '/': { | ||
|  |         'd': { | ||
|  |           action_: 'o=', | ||
|  |           nextState: '/' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}|else': { | ||
|  |         '0|d': { | ||
|  |           action_: 'd=', | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         'a': { | ||
|  |           action_: ['space', 'd='], | ||
|  |           nextState: 'd' | ||
|  |         }, | ||
|  |         '/|q': { | ||
|  |           action_: 'q=', | ||
|  |           nextState: 'q' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'enumber': function enumber(buffer, m) { | ||
|  |         /** @type {ParserOutput[]} */ | ||
|  |         var ret = []; | ||
|  | 
 | ||
|  |         if (m[0] === "+-" || m[0] === "+/-") { | ||
|  |           ret.push("\\pm "); | ||
|  |         } else if (m[0]) { | ||
|  |           ret.push(m[0]); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (m[1]) { | ||
|  |           mhchemParser.concatArray(ret, mhchemParser.go(m[1], 'pu-9,9')); | ||
|  | 
 | ||
|  |           if (m[2]) { | ||
|  |             if (m[2].match(/[,.]/)) { | ||
|  |               mhchemParser.concatArray(ret, mhchemParser.go(m[2], 'pu-9,9')); | ||
|  |             } else { | ||
|  |               ret.push(m[2]); | ||
|  |             } | ||
|  |           } | ||
|  | 
 | ||
|  |           m[3] = m[4] || m[3]; | ||
|  | 
 | ||
|  |           if (m[3]) { | ||
|  |             m[3] = m[3].trim(); | ||
|  | 
 | ||
|  |             if (m[3] === "e" || m[3].substr(0, 1) === "*") { | ||
|  |               ret.push({ | ||
|  |                 type_: 'cdot' | ||
|  |               }); | ||
|  |             } else { | ||
|  |               ret.push({ | ||
|  |                 type_: 'times' | ||
|  |               }); | ||
|  |             } | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         if (m[3]) { | ||
|  |           ret.push("10^{" + m[5] + "}"); | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'number^': function number(buffer, m) { | ||
|  |         /** @type {ParserOutput[]} */ | ||
|  |         var ret = []; | ||
|  | 
 | ||
|  |         if (m[0] === "+-" || m[0] === "+/-") { | ||
|  |           ret.push("\\pm "); | ||
|  |         } else if (m[0]) { | ||
|  |           ret.push(m[0]); | ||
|  |         } | ||
|  | 
 | ||
|  |         mhchemParser.concatArray(ret, mhchemParser.go(m[1], 'pu-9,9')); | ||
|  |         ret.push("^{" + m[2] + "}"); | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'operator': function operator(buffer, m, p1) { | ||
|  |         return { | ||
|  |           type_: 'operator', | ||
|  |           kind_: p1 || m | ||
|  |         }; | ||
|  |       }, | ||
|  |       'space': function space() { | ||
|  |         return { | ||
|  |           type_: 'pu-space-1' | ||
|  |         }; | ||
|  |       }, | ||
|  |       'output': function output(buffer) { | ||
|  |         /** @type {ParserOutput | ParserOutput[]} */ | ||
|  |         var ret; | ||
|  |         var md = mhchemParser.patterns.match_('{(...)}', buffer.d || ""); | ||
|  | 
 | ||
|  |         if (md && md.remainder === '') { | ||
|  |           buffer.d = md.match_; | ||
|  |         } | ||
|  | 
 | ||
|  |         var mq = mhchemParser.patterns.match_('{(...)}', buffer.q || ""); | ||
|  | 
 | ||
|  |         if (mq && mq.remainder === '') { | ||
|  |           buffer.q = mq.match_; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (buffer.d) { | ||
|  |           buffer.d = buffer.d.replace(/\u00B0C|\^oC|\^{o}C/g, "{}^{\\circ}C"); | ||
|  |           buffer.d = buffer.d.replace(/\u00B0F|\^oF|\^{o}F/g, "{}^{\\circ}F"); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (buffer.q) { | ||
|  |           // fraction
 | ||
|  |           buffer.q = buffer.q.replace(/\u00B0C|\^oC|\^{o}C/g, "{}^{\\circ}C"); | ||
|  |           buffer.q = buffer.q.replace(/\u00B0F|\^oF|\^{o}F/g, "{}^{\\circ}F"); | ||
|  |           var b5 = { | ||
|  |             d: mhchemParser.go(buffer.d, 'pu'), | ||
|  |             q: mhchemParser.go(buffer.q, 'pu') | ||
|  |           }; | ||
|  | 
 | ||
|  |           if (buffer.o === '//') { | ||
|  |             ret = { | ||
|  |               type_: 'pu-frac', | ||
|  |               p1: b5.d, | ||
|  |               p2: b5.q | ||
|  |             }; | ||
|  |           } else { | ||
|  |             ret = b5.d; | ||
|  | 
 | ||
|  |             if (b5.d.length > 1 || b5.q.length > 1) { | ||
|  |               ret.push({ | ||
|  |                 type_: ' / ' | ||
|  |               }); | ||
|  |             } else { | ||
|  |               ret.push({ | ||
|  |                 type_: '/' | ||
|  |               }); | ||
|  |             } | ||
|  | 
 | ||
|  |             mhchemParser.concatArray(ret, b5.q); | ||
|  |           } | ||
|  |         } else { | ||
|  |           // no fraction
 | ||
|  |           ret = mhchemParser.go(buffer.d, 'pu-2'); | ||
|  |         } | ||
|  | 
 | ||
|  |         for (var p in buffer) { | ||
|  |           delete buffer[p]; | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'pu-2': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '*': { | ||
|  |           action_: 'output' | ||
|  |         } | ||
|  |       }, | ||
|  |       '*': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'cdot'], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '\\x': { | ||
|  |         '*': { | ||
|  |           action_: 'rm=' | ||
|  |         } | ||
|  |       }, | ||
|  |       'space': { | ||
|  |         '*': { | ||
|  |           action_: ['output', 'space'], | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '^{(...)}|^(-1)': { | ||
|  |         '1': { | ||
|  |           action_: '^(-1)' | ||
|  |         } | ||
|  |       }, | ||
|  |       '-9.,9': { | ||
|  |         '0': { | ||
|  |           action_: 'rm=', | ||
|  |           nextState: '0' | ||
|  |         }, | ||
|  |         '1': { | ||
|  |           action_: '^(-1)', | ||
|  |           nextState: '0' | ||
|  |         } | ||
|  |       }, | ||
|  |       '{...}|else': { | ||
|  |         '*': { | ||
|  |           action_: 'rm=', | ||
|  |           nextState: '1' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'cdot': function cdot() { | ||
|  |         return { | ||
|  |           type_: 'tight cdot' | ||
|  |         }; | ||
|  |       }, | ||
|  |       '^(-1)': function _(buffer, m) { | ||
|  |         buffer.rm += "^{" + m + "}"; | ||
|  |       }, | ||
|  |       'space': function space() { | ||
|  |         return { | ||
|  |           type_: 'pu-space-2' | ||
|  |         }; | ||
|  |       }, | ||
|  |       'output': function output(buffer) { | ||
|  |         /** @type {ParserOutput | ParserOutput[]} */ | ||
|  |         var ret = []; | ||
|  | 
 | ||
|  |         if (buffer.rm) { | ||
|  |           var mrm = mhchemParser.patterns.match_('{(...)}', buffer.rm || ""); | ||
|  | 
 | ||
|  |           if (mrm && mrm.remainder === '') { | ||
|  |             ret = mhchemParser.go(mrm.match_, 'pu'); | ||
|  |           } else { | ||
|  |             ret = { | ||
|  |               type_: 'rm', | ||
|  |               p1: buffer.rm | ||
|  |             }; | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         for (var p in buffer) { | ||
|  |           delete buffer[p]; | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  |   'pu-9,9': { | ||
|  |     transitions: mhchemParser.createTransitions({ | ||
|  |       'empty': { | ||
|  |         '0': { | ||
|  |           action_: 'output-0' | ||
|  |         }, | ||
|  |         'o': { | ||
|  |           action_: 'output-o' | ||
|  |         } | ||
|  |       }, | ||
|  |       ',': { | ||
|  |         '0': { | ||
|  |           action_: ['output-0', 'comma'], | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       '.': { | ||
|  |         '0': { | ||
|  |           action_: ['output-0', 'copy'], | ||
|  |           nextState: 'o' | ||
|  |         } | ||
|  |       }, | ||
|  |       'else': { | ||
|  |         '*': { | ||
|  |           action_: 'text=' | ||
|  |         } | ||
|  |       } | ||
|  |     }), | ||
|  |     actions: { | ||
|  |       'comma': function comma() { | ||
|  |         return { | ||
|  |           type_: 'commaDecimal' | ||
|  |         }; | ||
|  |       }, | ||
|  |       'output-0': function output0(buffer) { | ||
|  |         /** @type {ParserOutput[]} */ | ||
|  |         var ret = []; | ||
|  |         buffer.text_ = buffer.text_ || ""; | ||
|  | 
 | ||
|  |         if (buffer.text_.length > 4) { | ||
|  |           var a = buffer.text_.length % 3; | ||
|  | 
 | ||
|  |           if (a === 0) { | ||
|  |             a = 3; | ||
|  |           } | ||
|  | 
 | ||
|  |           for (var i = buffer.text_.length - 3; i > 0; i -= 3) { | ||
|  |             ret.push(buffer.text_.substr(i, 3)); | ||
|  |             ret.push({ | ||
|  |               type_: '1000 separator' | ||
|  |             }); | ||
|  |           } | ||
|  | 
 | ||
|  |           ret.push(buffer.text_.substr(0, a)); | ||
|  |           ret.reverse(); | ||
|  |         } else { | ||
|  |           ret.push(buffer.text_); | ||
|  |         } | ||
|  | 
 | ||
|  |         for (var p in buffer) { | ||
|  |           delete buffer[p]; | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       }, | ||
|  |       'output-o': function outputO(buffer) { | ||
|  |         /** @type {ParserOutput[]} */ | ||
|  |         var ret = []; | ||
|  |         buffer.text_ = buffer.text_ || ""; | ||
|  | 
 | ||
|  |         if (buffer.text_.length > 4) { | ||
|  |           var a = buffer.text_.length - 3; | ||
|  | 
 | ||
|  |           for (var i = 0; i < a; i += 3) { | ||
|  |             ret.push(buffer.text_.substr(i, 3)); | ||
|  |             ret.push({ | ||
|  |               type_: '1000 separator' | ||
|  |             }); | ||
|  |           } | ||
|  | 
 | ||
|  |           ret.push(buffer.text_.substr(i)); | ||
|  |         } else { | ||
|  |           ret.push(buffer.text_); | ||
|  |         } | ||
|  | 
 | ||
|  |         for (var p in buffer) { | ||
|  |           delete buffer[p]; | ||
|  |         } | ||
|  | 
 | ||
|  |         return ret; | ||
|  |       } | ||
|  |     } | ||
|  |   } //#endregion
 | ||
|  | 
 | ||
|  | }; //
 | ||
|  | // texify: Take MhchemParser output and convert it to TeX
 | ||
|  | //
 | ||
|  | 
 | ||
|  | /** @type {Texify} */ | ||
|  | 
 | ||
|  | var texify = { | ||
|  |   go: function go(input, isInner) { | ||
|  |     // (recursive, max 4 levels)
 | ||
|  |     if (!input) { | ||
|  |       return ""; | ||
|  |     } | ||
|  | 
 | ||
|  |     var res = ""; | ||
|  |     var cee = false; | ||
|  | 
 | ||
|  |     for (var i = 0; i < input.length; i++) { | ||
|  |       var inputi = input[i]; | ||
|  | 
 | ||
|  |       if (typeof inputi === "string") { | ||
|  |         res += inputi; | ||
|  |       } else { | ||
|  |         res += texify._go2(inputi); | ||
|  | 
 | ||
|  |         if (inputi.type_ === '1st-level escape') { | ||
|  |           cee = true; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!isInner && !cee && res) { | ||
|  |       res = "{" + res + "}"; | ||
|  |     } | ||
|  | 
 | ||
|  |     return res; | ||
|  |   }, | ||
|  |   _goInner: function _goInner(input) { | ||
|  |     if (!input) { | ||
|  |       return input; | ||
|  |     } | ||
|  | 
 | ||
|  |     return texify.go(input, true); | ||
|  |   }, | ||
|  |   _go2: function _go2(buf) { | ||
|  |     /** @type {undefined | string} */ | ||
|  |     var res; | ||
|  | 
 | ||
|  |     switch (buf.type_) { | ||
|  |       case 'chemfive': | ||
|  |         res = ""; | ||
|  |         var b5 = { | ||
|  |           a: texify._goInner(buf.a), | ||
|  |           b: texify._goInner(buf.b), | ||
|  |           p: texify._goInner(buf.p), | ||
|  |           o: texify._goInner(buf.o), | ||
|  |           q: texify._goInner(buf.q), | ||
|  |           d: texify._goInner(buf.d) | ||
|  |         }; //
 | ||
|  |         // a
 | ||
|  |         //
 | ||
|  | 
 | ||
|  |         if (b5.a) { | ||
|  |           if (b5.a.match(/^[+\-]/)) { | ||
|  |             b5.a = "{" + b5.a + "}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           res += b5.a + "\\,"; | ||
|  |         } //
 | ||
|  |         // b and p
 | ||
|  |         //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         if (b5.b || b5.p) { | ||
|  |           res += "{\\vphantom{X}}"; | ||
|  |           res += "^{\\hphantom{" + (b5.b || "") + "}}_{\\hphantom{" + (b5.p || "") + "}}"; | ||
|  |           res += "{\\vphantom{X}}"; | ||
|  |           res += "^{\\smash[t]{\\vphantom{2}}\\mathllap{" + (b5.b || "") + "}}"; | ||
|  |           res += "_{\\vphantom{2}\\mathllap{\\smash[t]{" + (b5.p || "") + "}}}"; | ||
|  |         } //
 | ||
|  |         // o
 | ||
|  |         //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         if (b5.o) { | ||
|  |           if (b5.o.match(/^[+\-]/)) { | ||
|  |             b5.o = "{" + b5.o + "}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           res += b5.o; | ||
|  |         } //
 | ||
|  |         // q and d
 | ||
|  |         //
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         if (buf.dType === 'kv') { | ||
|  |           if (b5.d || b5.q) { | ||
|  |             res += "{\\vphantom{X}}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           if (b5.d) { | ||
|  |             res += "^{" + b5.d + "}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           if (b5.q) { | ||
|  |             res += "_{\\smash[t]{" + b5.q + "}}"; | ||
|  |           } | ||
|  |         } else if (buf.dType === 'oxidation') { | ||
|  |           if (b5.d) { | ||
|  |             res += "{\\vphantom{X}}"; | ||
|  |             res += "^{" + b5.d + "}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           if (b5.q) { | ||
|  |             res += "{\\vphantom{X}}"; | ||
|  |             res += "_{\\smash[t]{" + b5.q + "}}"; | ||
|  |           } | ||
|  |         } else { | ||
|  |           if (b5.q) { | ||
|  |             res += "{\\vphantom{X}}"; | ||
|  |             res += "_{\\smash[t]{" + b5.q + "}}"; | ||
|  |           } | ||
|  | 
 | ||
|  |           if (b5.d) { | ||
|  |             res += "{\\vphantom{X}}"; | ||
|  |             res += "^{" + b5.d + "}"; | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'rm': | ||
|  |         res = "\\mathrm{" + buf.p1 + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'text': | ||
|  |         if (buf.p1.match(/[\^_]/)) { | ||
|  |           buf.p1 = buf.p1.replace(" ", "~").replace("-", "\\text{-}"); | ||
|  |           res = "\\mathrm{" + buf.p1 + "}"; | ||
|  |         } else { | ||
|  |           res = "\\text{" + buf.p1 + "}"; | ||
|  |         } | ||
|  | 
 | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'roman numeral': | ||
|  |         res = "\\mathrm{" + buf.p1 + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'state of aggregation': | ||
|  |         res = "\\mskip2mu " + texify._goInner(buf.p1); | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'state of aggregation subscript': | ||
|  |         res = "\\mskip1mu " + texify._goInner(buf.p1); | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'bond': | ||
|  |         res = texify._getBond(buf.kind_); | ||
|  | 
 | ||
|  |         if (!res) { | ||
|  |           throw ["MhchemErrorBond", "mhchem Error. Unknown bond type (" + buf.kind_ + ")"]; | ||
|  |         } | ||
|  | 
 | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'frac': | ||
|  |         var c = "\\frac{" + buf.p1 + "}{" + buf.p2 + "}"; | ||
|  |         res = "\\mathchoice{\\textstyle" + c + "}{" + c + "}{" + c + "}{" + c + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'pu-frac': | ||
|  |         var d = "\\frac{" + texify._goInner(buf.p1) + "}{" + texify._goInner(buf.p2) + "}"; | ||
|  |         res = "\\mathchoice{\\textstyle" + d + "}{" + d + "}{" + d + "}{" + d + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'tex-math': | ||
|  |         res = buf.p1 + " "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'frac-ce': | ||
|  |         res = "\\frac{" + texify._goInner(buf.p1) + "}{" + texify._goInner(buf.p2) + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'overset': | ||
|  |         res = "\\overset{" + texify._goInner(buf.p1) + "}{" + texify._goInner(buf.p2) + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'underset': | ||
|  |         res = "\\underset{" + texify._goInner(buf.p1) + "}{" + texify._goInner(buf.p2) + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'underbrace': | ||
|  |         res = "\\underbrace{" + texify._goInner(buf.p1) + "}_{" + texify._goInner(buf.p2) + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'color': | ||
|  |         res = "{\\color{" + buf.color1 + "}{" + texify._goInner(buf.color2) + "}}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'color0': | ||
|  |         res = "\\color{" + buf.color + "}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'arrow': | ||
|  |         var b6 = { | ||
|  |           rd: texify._goInner(buf.rd), | ||
|  |           rq: texify._goInner(buf.rq) | ||
|  |         }; | ||
|  | 
 | ||
|  |         var arrow = "\\x" + texify._getArrow(buf.r); | ||
|  | 
 | ||
|  |         if (b6.rq) { | ||
|  |           arrow += "[{" + b6.rq + "}]"; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (b6.rd) { | ||
|  |           arrow += "{" + b6.rd + "}"; | ||
|  |         } else { | ||
|  |           arrow += "{}"; | ||
|  |         } | ||
|  | 
 | ||
|  |         res = arrow; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'operator': | ||
|  |         res = texify._getOperator(buf.kind_); | ||
|  |         break; | ||
|  | 
 | ||
|  |       case '1st-level escape': | ||
|  |         res = buf.p1 + " "; // &, \\\\, \\hlin
 | ||
|  | 
 | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'space': | ||
|  |         res = " "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'entitySkip': | ||
|  |         res = "~"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'pu-space-1': | ||
|  |         res = "~"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'pu-space-2': | ||
|  |         res = "\\mkern3mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case '1000 separator': | ||
|  |         res = "\\mkern2mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'commaDecimal': | ||
|  |         res = "{,}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'comma enumeration L': | ||
|  |         res = "{" + buf.p1 + "}\\mkern6mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'comma enumeration M': | ||
|  |         res = "{" + buf.p1 + "}\\mkern3mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'comma enumeration S': | ||
|  |         res = "{" + buf.p1 + "}\\mkern1mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'hyphen': | ||
|  |         res = "\\text{-}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'addition compound': | ||
|  |         res = "\\,{\\cdot}\\,"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'electron dot': | ||
|  |         res = "\\mkern1mu \\bullet\\mkern1mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'KV x': | ||
|  |         res = "{\\times}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'prime': | ||
|  |         res = "\\prime "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'cdot': | ||
|  |         res = "\\cdot "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'tight cdot': | ||
|  |         res = "\\mkern1mu{\\cdot}\\mkern1mu "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'times': | ||
|  |         res = "\\times "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'circa': | ||
|  |         res = "{\\sim}"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case '^': | ||
|  |         res = "uparrow"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'v': | ||
|  |         res = "downarrow"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case 'ellipsis': | ||
|  |         res = "\\ldots "; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case '/': | ||
|  |         res = "/"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case ' / ': | ||
|  |         res = "\\,/\\,"; | ||
|  |         break; | ||
|  | 
 | ||
|  |       default: | ||
|  |         throw ["MhchemBugT", "mhchem bug T. Please report."]; | ||
|  |       // Missing texify rule or unknown MhchemParser output
 | ||
|  |     } | ||
|  |     return res; | ||
|  |   }, | ||
|  |   _getArrow: function _getArrow(a) { | ||
|  |     switch (a) { | ||
|  |       case "->": | ||
|  |         return "rightarrow"; | ||
|  | 
 | ||
|  |       case "\u2192": | ||
|  |         return "rightarrow"; | ||
|  | 
 | ||
|  |       case "\u27F6": | ||
|  |         return "rightarrow"; | ||
|  | 
 | ||
|  |       case "<-": | ||
|  |         return "leftarrow"; | ||
|  | 
 | ||
|  |       case "<->": | ||
|  |         return "leftrightarrow"; | ||
|  | 
 | ||
|  |       case "<-->": | ||
|  |         return "rightleftarrows"; | ||
|  | 
 | ||
|  |       case "<=>": | ||
|  |         return "rightleftharpoons"; | ||
|  | 
 | ||
|  |       case "\u21CC": | ||
|  |         return "rightleftharpoons"; | ||
|  | 
 | ||
|  |       case "<=>>": | ||
|  |         return "rightequilibrium"; | ||
|  | 
 | ||
|  |       case "<<=>": | ||
|  |         return "leftequilibrium"; | ||
|  | 
 | ||
|  |       default: | ||
|  |         throw ["MhchemBugT", "mhchem bug T. Please report."]; | ||
|  |     } | ||
|  |   }, | ||
|  |   _getBond: function _getBond(a) { | ||
|  |     switch (a) { | ||
|  |       case "-": | ||
|  |         return "{-}"; | ||
|  | 
 | ||
|  |       case "1": | ||
|  |         return "{-}"; | ||
|  | 
 | ||
|  |       case "=": | ||
|  |         return "{=}"; | ||
|  | 
 | ||
|  |       case "2": | ||
|  |         return "{=}"; | ||
|  | 
 | ||
|  |       case "#": | ||
|  |         return "{\\equiv}"; | ||
|  | 
 | ||
|  |       case "3": | ||
|  |         return "{\\equiv}"; | ||
|  | 
 | ||
|  |       case "~": | ||
|  |         return "{\\tripledash}"; | ||
|  | 
 | ||
|  |       case "~-": | ||
|  |         return "{\\mathrlap{\\raisebox{-.1em}{$-$}}\\raisebox{.1em}{$\\tripledash$}}"; | ||
|  | 
 | ||
|  |       case "~=": | ||
|  |         return "{\\mathrlap{\\raisebox{-.2em}{$-$}}\\mathrlap{\\raisebox{.2em}{$\\tripledash$}}-}"; | ||
|  | 
 | ||
|  |       case "~--": | ||
|  |         return "{\\mathrlap{\\raisebox{-.2em}{$-$}}\\mathrlap{\\raisebox{.2em}{$\\tripledash$}}-}"; | ||
|  | 
 | ||
|  |       case "-~-": | ||
|  |         return "{\\mathrlap{\\raisebox{-.2em}{$-$}}\\mathrlap{\\raisebox{.2em}{$-$}}\\tripledash}"; | ||
|  | 
 | ||
|  |       case "...": | ||
|  |         return "{{\\cdot}{\\cdot}{\\cdot}}"; | ||
|  | 
 | ||
|  |       case "....": | ||
|  |         return "{{\\cdot}{\\cdot}{\\cdot}{\\cdot}}"; | ||
|  | 
 | ||
|  |       case "->": | ||
|  |         return "{\\rightarrow}"; | ||
|  | 
 | ||
|  |       case "<-": | ||
|  |         return "{\\leftarrow}"; | ||
|  | 
 | ||
|  |       case "<": | ||
|  |         return "{<}"; | ||
|  | 
 | ||
|  |       case ">": | ||
|  |         return "{>}"; | ||
|  | 
 | ||
|  |       default: | ||
|  |         throw ["MhchemBugT", "mhchem bug T. Please report."]; | ||
|  |     } | ||
|  |   }, | ||
|  |   _getOperator: function _getOperator(a) { | ||
|  |     switch (a) { | ||
|  |       case "+": | ||
|  |         return " {}+{} "; | ||
|  | 
 | ||
|  |       case "-": | ||
|  |         return " {}-{} "; | ||
|  | 
 | ||
|  |       case "=": | ||
|  |         return " {}={} "; | ||
|  | 
 | ||
|  |       case "<": | ||
|  |         return " {}<{} "; | ||
|  | 
 | ||
|  |       case ">": | ||
|  |         return " {}>{} "; | ||
|  | 
 | ||
|  |       case "<<": | ||
|  |         return " {}\\ll{} "; | ||
|  | 
 | ||
|  |       case ">>": | ||
|  |         return " {}\\gg{} "; | ||
|  | 
 | ||
|  |       case "\\pm": | ||
|  |         return " {}\\pm{} "; | ||
|  | 
 | ||
|  |       case "\\approx": | ||
|  |         return " {}\\approx{} "; | ||
|  | 
 | ||
|  |       case "$\\approx$": | ||
|  |         return " {}\\approx{} "; | ||
|  | 
 | ||
|  |       case "v": | ||
|  |         return " \\downarrow{} "; | ||
|  | 
 | ||
|  |       case "(v)": | ||
|  |         return " \\downarrow{} "; | ||
|  | 
 | ||
|  |       case "^": | ||
|  |         return " \\uparrow{} "; | ||
|  | 
 | ||
|  |       case "(^)": | ||
|  |         return " \\uparrow{} "; | ||
|  | 
 | ||
|  |       default: | ||
|  |         throw ["MhchemBugT", "mhchem bug T. Please report."]; | ||
|  |     } | ||
|  |   } | ||
|  | }; //
 |