(function (Prism) { // https://freemarker.apache.org/docs/dgui_template_exp.html // FTL expression with 4 levels of nesting supported var FTL_EXPR = /[^<()"']|\((?:)*\)|<(?!#--)|<#--(?:[^-]|-(?!->))*-->|"(?:[^\\"]|\\.)*"|'(?:[^\\']|\\.)*'/ .source; for (var i = 0; i < 2; i++) { FTL_EXPR = FTL_EXPR.replace(//g, function () { return FTL_EXPR; }); } FTL_EXPR = FTL_EXPR.replace(//g, /[^\s\S]/.source); var ftl = { comment: /<#--[\s\S]*?-->/, string: [ { // raw string pattern: /\br("|')(?:(?!\1)[^\\]|\\.)*\1/, greedy: true, }, { pattern: RegExp( /("|')(?:(?!\1|\$\{)[^\\]|\\.|\$\{(?:(?!\})(?:))*\})*\1/.source.replace( //g, function () { return FTL_EXPR; } ) ), greedy: true, inside: { interpolation: { pattern: RegExp( /((?:^|[^\\])(?:\\\\)*)\$\{(?:(?!\})(?:))*\}/.source.replace( //g, function () { return FTL_EXPR; } ) ), lookbehind: true, inside: { 'interpolation-punctuation': { pattern: /^\$\{|\}$/, alias: 'punctuation', }, rest: null, }, }, }, }, ], keyword: /\b(?:as)\b/, boolean: /\b(?:false|true)\b/, 'builtin-function': { pattern: /((?:^|[^?])\?\s*)\w+/, lookbehind: true, alias: 'function', }, function: /\b\w+(?=\s*\()/, number: /\b\d+(?:\.\d+)?\b/, operator: /\.\.[<*!]?|->|--|\+\+|&&|\|\||\?{1,2}|[-+*/%!=<>]=?|\b(?:gt|gte|lt|lte)\b/, punctuation: /[,;.:()[\]{}]/, }; ftl.string[1].inside.interpolation.inside.rest = ftl; Prism.languages.ftl = { 'ftl-comment': { // the pattern is shortened to be more efficient pattern: /^<#--[\s\S]*/, alias: 'comment', }, 'ftl-directive': { pattern: /^<[\s\S]+>$/, inside: { directive: { pattern: /(^<\/?)[#@][a-z]\w*/i, lookbehind: true, alias: 'keyword', }, punctuation: /^<\/?|\/?>$/, content: { pattern: /\s*\S[\s\S]*/, alias: 'ftl', inside: ftl, }, }, }, 'ftl-interpolation': { pattern: /^\$\{[\s\S]*\}$/, inside: { punctuation: /^\$\{|\}$/, content: { pattern: /\s*\S[\s\S]*/, alias: 'ftl', inside: ftl, }, }, }, }; Prism.hooks.add('before-tokenize', function (env) { // eslint-disable-next-line regexp/no-useless-lazy var pattern = RegExp( /<#--[\s\S]*?-->|<\/?[#@][a-zA-Z](?:)*?>|\$\{(?:)*?\}/.source.replace( //g, function () { return FTL_EXPR; } ), 'gi' ); Prism.languages['markup-templating'].buildPlaceholders(env, 'ftl', pattern); }); Prism.hooks.add('after-tokenize', function (env) { Prism.languages['markup-templating'].tokenizePlaceholders(env, 'ftl'); }); })(Prism);