277 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * @param {string} value
 | |
|  * @returns {RegExp}
 | |
|  * */
 | |
| 
 | |
| /**
 | |
|  * @param {RegExp | string } re
 | |
|  * @returns {string}
 | |
|  */
 | |
| function source(re) {
 | |
|   if (!re) return null;
 | |
|   if (typeof re === "string") return re;
 | |
| 
 | |
|   return re.source;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Any of the passed expresssions may match
 | |
|  *
 | |
|  * Creates a huge this | this | that | that match
 | |
|  * @param {(RegExp | string)[] } args
 | |
|  * @returns {string}
 | |
|  */
 | |
| function either(...args) {
 | |
|   const joined = '(' + args.map((x) => source(x)).join("|") + ")";
 | |
|   return joined;
 | |
| }
 | |
| 
 | |
| /*
 | |
| Language: LaTeX
 | |
| Author: Benedikt Wilde <bwilde@posteo.de>
 | |
| Website: https://www.latex-project.org
 | |
| Category: markup
 | |
| */
 | |
| 
 | |
| /** @type LanguageFn */
 | |
| function latex(hljs) {
 | |
|   const KNOWN_CONTROL_WORDS = either(...[
 | |
|       '(?:NeedsTeXFormat|RequirePackage|GetIdInfo)',
 | |
|       'Provides(?:Expl)?(?:Package|Class|File)',
 | |
|       '(?:DeclareOption|ProcessOptions)',
 | |
|       '(?:documentclass|usepackage|input|include)',
 | |
|       'makeat(?:letter|other)',
 | |
|       'ExplSyntax(?:On|Off)',
 | |
|       '(?:new|renew|provide)?command',
 | |
|       '(?:re)newenvironment',
 | |
|       '(?:New|Renew|Provide|Declare)(?:Expandable)?DocumentCommand',
 | |
|       '(?:New|Renew|Provide|Declare)DocumentEnvironment',
 | |
|       '(?:(?:e|g|x)?def|let)',
 | |
|       '(?:begin|end)',
 | |
|       '(?:part|chapter|(?:sub){0,2}section|(?:sub)?paragraph)',
 | |
|       'caption',
 | |
|       '(?:label|(?:eq|page|name)?ref|(?:paren|foot|super)?cite)',
 | |
|       '(?:alpha|beta|[Gg]amma|[Dd]elta|(?:var)?epsilon|zeta|eta|[Tt]heta|vartheta)',
 | |
|       '(?:iota|(?:var)?kappa|[Ll]ambda|mu|nu|[Xx]i|[Pp]i|varpi|(?:var)rho)',
 | |
|       '(?:[Ss]igma|varsigma|tau|[Uu]psilon|[Pp]hi|varphi|chi|[Pp]si|[Oo]mega)',
 | |
|       '(?:frac|sum|prod|lim|infty|times|sqrt|leq|geq|left|right|middle|[bB]igg?)',
 | |
|       '(?:[lr]angle|q?quad|[lcvdi]?dots|d?dot|hat|tilde|bar)'
 | |
|     ].map(word => word + '(?![a-zA-Z@:_])'));
 | |
|   const L3_REGEX = new RegExp([
 | |
|       // A function \module_function_name:signature or \__module_function_name:signature,
 | |
|       // where both module and function_name need at least two characters and
 | |
|       // function_name may contain single underscores.
 | |
|       '(?:__)?[a-zA-Z]{2,}_[a-zA-Z](?:_?[a-zA-Z])+:[a-zA-Z]*',
 | |
|       // A variable \scope_module_and_name_type or \scope__module_ane_name_type,
 | |
|       // where scope is one of l, g or c, type needs at least two characters
 | |
|       // and module_and_name may contain single underscores.
 | |
|       '[lgc]__?[a-zA-Z](?:_?[a-zA-Z])*_[a-zA-Z]{2,}',
 | |
|       // A quark \q_the_name or \q__the_name or
 | |
|       // scan mark \s_the_name or \s__vthe_name,
 | |
|       // where variable_name needs at least two characters and
 | |
|       // may contain single underscores.
 | |
|       '[qs]__?[a-zA-Z](?:_?[a-zA-Z])+',
 | |
|       // Other LaTeX3 macro names that are not covered by the three rules above.
 | |
|       'use(?:_i)?:[a-zA-Z]*',
 | |
|       '(?:else|fi|or):',
 | |
|       '(?:if|cs|exp):w',
 | |
|       '(?:hbox|vbox):n',
 | |
|       '::[a-zA-Z]_unbraced',
 | |
|       '::[a-zA-Z:]'
 | |
|     ].map(pattern => pattern + '(?![a-zA-Z:_])').join('|'));
 | |
|   const L2_VARIANTS = [
 | |
|     {begin: /[a-zA-Z@]+/}, // control word
 | |
|     {begin: /[^a-zA-Z@]?/} // control symbol
 | |
|   ];
 | |
|   const DOUBLE_CARET_VARIANTS = [
 | |
|     {begin: /\^{6}[0-9a-f]{6}/},
 | |
|     {begin: /\^{5}[0-9a-f]{5}/},
 | |
|     {begin: /\^{4}[0-9a-f]{4}/},
 | |
|     {begin: /\^{3}[0-9a-f]{3}/},
 | |
|     {begin: /\^{2}[0-9a-f]{2}/},
 | |
|     {begin: /\^{2}[\u0000-\u007f]/}
 | |
|   ];
 | |
|   const CONTROL_SEQUENCE = {
 | |
|     className: 'keyword',
 | |
|     begin: /\\/,
 | |
|     relevance: 0,
 | |
|     contains: [
 | |
|       {
 | |
|         endsParent: true,
 | |
|         begin: KNOWN_CONTROL_WORDS
 | |
|       },
 | |
|       {
 | |
|         endsParent: true,
 | |
|         begin: L3_REGEX
 | |
|       },
 | |
|       {
 | |
|         endsParent: true,
 | |
|         variants: DOUBLE_CARET_VARIANTS
 | |
|       },
 | |
|       {
 | |
|         endsParent: true,
 | |
|         relevance: 0,
 | |
|         variants: L2_VARIANTS
 | |
|       }
 | |
|     ]
 | |
|   };
 | |
|   const MACRO_PARAM = {
 | |
|     className: 'params',
 | |
|     relevance: 0,
 | |
|     begin: /#+\d?/
 | |
|   };
 | |
|   const DOUBLE_CARET_CHAR = {
 | |
|     // relevance: 1
 | |
|     variants: DOUBLE_CARET_VARIANTS
 | |
|   };
 | |
|   const SPECIAL_CATCODE = {
 | |
|     className: 'built_in',
 | |
|     relevance: 0,
 | |
|     begin: /[$&^_]/
 | |
|   };
 | |
|   const MAGIC_COMMENT = {
 | |
|     className: 'meta',
 | |
|     begin: '% !TeX',
 | |
|     end: '$',
 | |
|     relevance: 10
 | |
|   };
 | |
|   const COMMENT = hljs.COMMENT(
 | |
|     '%',
 | |
|     '$',
 | |
|     {
 | |
|       relevance: 0
 | |
|     }
 | |
|   );
 | |
|   const EVERYTHING_BUT_VERBATIM = [
 | |
|     CONTROL_SEQUENCE,
 | |
|     MACRO_PARAM,
 | |
|     DOUBLE_CARET_CHAR,
 | |
|     SPECIAL_CATCODE,
 | |
|     MAGIC_COMMENT,
 | |
|     COMMENT
 | |
|   ];
 | |
|   const BRACE_GROUP_NO_VERBATIM = {
 | |
|     begin: /\{/, end: /\}/,
 | |
|     relevance: 0,
 | |
|     contains: ['self', ...EVERYTHING_BUT_VERBATIM]
 | |
|   };
 | |
|   const ARGUMENT_BRACES = hljs.inherit(
 | |
|     BRACE_GROUP_NO_VERBATIM,
 | |
|     {
 | |
|       relevance: 0,
 | |
|       endsParent: true,
 | |
|       contains: [BRACE_GROUP_NO_VERBATIM, ...EVERYTHING_BUT_VERBATIM]
 | |
|     }
 | |
|   );
 | |
|   const ARGUMENT_BRACKETS = {
 | |
|     begin: /\[/,
 | |
|       end: /\]/,
 | |
|     endsParent: true,
 | |
|     relevance: 0,
 | |
|     contains: [BRACE_GROUP_NO_VERBATIM, ...EVERYTHING_BUT_VERBATIM]
 | |
|   };
 | |
|   const SPACE_GOBBLER = {
 | |
|     begin: /\s+/,
 | |
|     relevance: 0
 | |
|   };
 | |
|   const ARGUMENT_M = [ARGUMENT_BRACES];
 | |
|   const ARGUMENT_O = [ARGUMENT_BRACKETS];
 | |
|   const ARGUMENT_AND_THEN = function(arg, starts_mode) {
 | |
|     return {
 | |
|       contains: [SPACE_GOBBLER],
 | |
|       starts: {
 | |
|         relevance: 0,
 | |
|         contains: arg,
 | |
|         starts: starts_mode
 | |
|       }
 | |
|     };
 | |
|   };
 | |
|   const CSNAME = function(csname, starts_mode) {
 | |
|     return {
 | |
|         begin: '\\\\' + csname + '(?![a-zA-Z@:_])',
 | |
|         keywords: {$pattern: /\\[a-zA-Z]+/, keyword: '\\' + csname},
 | |
|         relevance: 0,
 | |
|         contains: [SPACE_GOBBLER],
 | |
|         starts: starts_mode
 | |
|       };
 | |
|   };
 | |
|   const BEGIN_ENV = function(envname, starts_mode) {
 | |
|     return hljs.inherit(
 | |
|       {
 | |
|         begin: '\\\\begin(?=[ \t]*(\\r?\\n[ \t]*)?\\{' + envname + '\\})',
 | |
|         keywords: {$pattern: /\\[a-zA-Z]+/, keyword: '\\begin'},
 | |
|         relevance: 0,
 | |
|       },
 | |
|       ARGUMENT_AND_THEN(ARGUMENT_M, starts_mode)
 | |
|     );
 | |
|   };
 | |
|   const VERBATIM_DELIMITED_EQUAL = (innerName = "string") => {
 | |
|     return hljs.END_SAME_AS_BEGIN({
 | |
|       className: innerName,
 | |
|       begin: /(.|\r?\n)/,
 | |
|       end: /(.|\r?\n)/,
 | |
|       excludeBegin: true,
 | |
|       excludeEnd: true,
 | |
|       endsParent: true
 | |
|     });
 | |
|   };
 | |
|   const VERBATIM_DELIMITED_ENV = function(envname) {
 | |
|     return {
 | |
|       className: 'string',
 | |
|       end: '(?=\\\\end\\{' + envname + '\\})'
 | |
|     };
 | |
|   };
 | |
| 
 | |
|   const VERBATIM_DELIMITED_BRACES = (innerName = "string") => {
 | |
|     return {
 | |
|       relevance: 0,
 | |
|       begin: /\{/,
 | |
|       starts: {
 | |
|         endsParent: true,
 | |
|         contains: [
 | |
|           {
 | |
|             className: innerName,
 | |
|             end: /(?=\})/,
 | |
|             endsParent:true,
 | |
|             contains: [
 | |
|               {
 | |
|                 begin: /\{/,
 | |
|                 end: /\}/,
 | |
|                 relevance: 0,
 | |
|                 contains: ["self"]
 | |
|               }
 | |
|             ],
 | |
|           }
 | |
|         ]
 | |
|       }
 | |
|     };
 | |
|   };
 | |
|   const VERBATIM = [
 | |
|     ...['verb', 'lstinline'].map(csname => CSNAME(csname, {contains: [VERBATIM_DELIMITED_EQUAL()]})),
 | |
|     CSNAME('mint', ARGUMENT_AND_THEN(ARGUMENT_M, {contains: [VERBATIM_DELIMITED_EQUAL()]})),
 | |
|     CSNAME('mintinline', ARGUMENT_AND_THEN(ARGUMENT_M, {contains: [VERBATIM_DELIMITED_BRACES(), VERBATIM_DELIMITED_EQUAL()]})),
 | |
|     CSNAME('url', {contains: [VERBATIM_DELIMITED_BRACES("link"), VERBATIM_DELIMITED_BRACES("link")]}),
 | |
|     CSNAME('hyperref', {contains: [VERBATIM_DELIMITED_BRACES("link")]}),
 | |
|     CSNAME('href', ARGUMENT_AND_THEN(ARGUMENT_O, {contains: [VERBATIM_DELIMITED_BRACES("link")]})),
 | |
|     ...[].concat(...['', '\\*'].map(suffix => [
 | |
|       BEGIN_ENV('verbatim' + suffix, VERBATIM_DELIMITED_ENV('verbatim' + suffix)),
 | |
|       BEGIN_ENV('filecontents' + suffix,  ARGUMENT_AND_THEN(ARGUMENT_M, VERBATIM_DELIMITED_ENV('filecontents' + suffix))),
 | |
|       ...['', 'B', 'L'].map(prefix =>
 | |
|         BEGIN_ENV(prefix + 'Verbatim' + suffix, ARGUMENT_AND_THEN(ARGUMENT_O, VERBATIM_DELIMITED_ENV(prefix + 'Verbatim' + suffix)))
 | |
|       )
 | |
|     ])),
 | |
|     BEGIN_ENV('minted', ARGUMENT_AND_THEN(ARGUMENT_O, ARGUMENT_AND_THEN(ARGUMENT_M, VERBATIM_DELIMITED_ENV('minted')))),
 | |
|   ];
 | |
| 
 | |
|   return {
 | |
|     name: 'LaTeX',
 | |
|     aliases: ['tex'],
 | |
|     contains: [
 | |
|       ...VERBATIM,
 | |
|       ...EVERYTHING_BUT_VERBATIM
 | |
|     ]
 | |
|   };
 | |
| }
 | |
| 
 | |
| module.exports = latex;
 |