mirror of
				https://github.com/TriliumNext/Notes.git
				synced 2025-10-26 01:21: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.length && 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."];
 | |
|     }
 | |
|   }
 | |
| }; //
 | 
