aboutsummaryrefslogtreecommitdiffstats
path: root/public/prism/prism-naniscript.js
blob: 46c233201029c877480e759bd1aa0e7f957b58a8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
pre { line-height: 125%; }
td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
.highlight .hll { background-color: #ffffcc }
.highlight .c { color: #888888 } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { color: #008800; font-weight: bold } /* Keyword */
.highlight .ch { color: #888888 } /* Comment.Hashbang */
.highlight .cm { color: #888888 } /* Comment.Multiline */
.highlight .cp { color: #cc0000; font-weight: bold } /* Comment.Preproc */
.highlight .cpf { color: #888888 } /* Comment.PreprocFile */
.highlight .c1 { color: #888888 } /* Comment.Single */
.highlight .cs { color: #cc0000; font-weight: bold; background-color: #fff0f0 } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #333333 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #666666 } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { color: #008800; font-weight: bold } /* Keyword.Constant */
.highlight .kd { color: #008800; font-weight: bold } /* Keyword.Declaration */
.highlight .kn { color: #008800; font-weight: bold } /* Keyword.Namespace */
.highlight .kp { color: #008800 } /* Keyword.Pseudo */
.highlight .kr { color: #008800; font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #888888; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #0000DD; font-weight: bold } /* Literal.Number */
.highlight .s { color: #dd2200; background-color: #fff0f0 } /* Literal.String */
.highlight .na { color: #336699 } /* Name.Attribute */
.highlight .nb { color: #003388 } /* Name.Builtin */
.highlight .nc { color: #bb0066; font-weight: bold } /* Name.Class */
.highlight .no { color: #003366; font-weight: bold } /* Name.Constant */
.highlight .nd { color: #555555 } /* Name.Decorator */
.highlight .ne { color: #bb0066; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #0066bb; font-weight: bold } /* Name.Function */
.highlight .nl { color: #336699; font-style: italic } /* Name.Label */
.highlight .nn { color: #bb0066; font-weight: bold } /* Name.Namespace */
.highlight .py { color: #336699; font-weight: bold } /* Name.Property */
.highlight .nt { color: #bb0066; font-weight: bold } /* Name.Tag */
.highlight .nv { color: #336699 } /* Name.Variable */
.highlight .ow { color: #008800 } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mb { color: #0000DD; font-weight: bold } /* Literal.Number.Bin */
.highlight .mf { color: #0000DD; font-weight: bold } /* Literal.Number.Float */
.highlight .mh { color: #0000DD; font-weight: bold } /* Literal.Number.Hex */
.highlight .mi { color: #0000DD; font-weight: bold } /* Literal.Number.Integer */
.highlight .mo { color: #0000DD; font-weight: bold } /* Literal.Number.Oct */
.highlight .sa { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Affix */
.highlight .sb { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Backtick */
.highlight .sc { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Char */
.highlight .dl { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Delimiter */
.highlight .sd { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Doc */
.highlight .s2 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Double */
.highlight .se { color: #0044dd; background-color: #fff0f0 } /* Literal.String.Escape */
.highlight .sh { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Heredoc */
.highlight .si { color: #3333bb; background-color: #fff0f0 } /* Literal.String.Interpol */
.highlight .sx { color: #22bb22; background-color: #f0fff0 } /* Literal.String.Other */
.highlight .sr { color: #008800; background-color: #fff0ff } /* Literal.String.Regex */
.highlight .s1 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Single */
.highlight .ss { color: #aa6600; background-color: #fff0f0 } /* Literal.String.Symbol */
.highlight .bp { color: #003388 } /* Name.Builtin.Pseudo */
.highlight .fm { color: #0066bb; font-weight: bold } /* Name.Function
(function (Prism) {
  var expressionDef = /\{[^\r\n\[\]{}]*\}/;

  var params = {
    'quoted-string': {
      pattern: /"(?:[^"\\]|\\.)*"/,
      alias: 'operator',
    },
    'command-param-id': {
      pattern: /(\s)\w+:/,
      lookbehind: true,
      alias: 'property',
    },
    'command-param-value': [
      {
        pattern: expressionDef,
        alias: 'selector',
      },
      {
        pattern: /([\t ])\S+/,
        lookbehind: true,
        greedy: true,
        alias: 'operator',
      },
      {
        pattern: /\S(?:.*\S)?/,
        alias: 'operator',
      },
    ],
  };

  Prism.languages.naniscript = {
    // ; ...
    comment: {
      pattern: /^([\t ]*);.*/m,
      lookbehind: true,
    },
    // > ...
    // Define is a control line starting with '>' followed by a word, a space and a text.
    define: {
      pattern: /^>.+/m,
      alias: 'tag',
      inside: {
        value: {
          pattern: /(^>\w+[\t ]+)(?!\s)[^{}\r\n]+/,
          lookbehind: true,
          alias: 'operator',
        },
        key: {
          pattern: /(^>)\w+/,
          lookbehind: true,
        },
      },
    },
    // # ...
    label: {
      pattern: /^([\t ]*)#[\t ]*\w+[\t ]*$/m,
      lookbehind: true,
      alias: 'regex',
    },
    command: {
      pattern: /^([\t ]*)@\w+(?=[\t ]|$).*/m,
      lookbehind: true,
      alias: 'function',
      inside: {
        'command-name': /^@\w+/,
        expression: {
          pattern: expressionDef,
          greedy: true,
          alias: 'selector',
        },
        'command-params': {
          pattern: /\s*\S[\s\S]*/,
          inside: params,
        },
      },
    },
    // Generic is any line that doesn't start with operators: ;>#@
    'generic-text': {
      pattern: /(^[ \t]*)[^#@>;\s].*/m,
      lookbehind: true,
      alias: 'punctuation',
      inside: {
        // \{ ... \} ... \[ ... \] ... \"
        'escaped-char': /\\[{}\[\]"]/,
        expression: {
          pattern: expressionDef,
          greedy: true,
          alias: 'selector',
        },
        'inline-command': {
          pattern: /\[[\t ]*\w[^\r\n\[\]]*\]/,
          greedy: true,
          alias: 'function',
          inside: {
            'command-params': {
              pattern: /(^\[[\t ]*\w+\b)[\s\S]+(?=\]$)/,
              lookbehind: true,
              inside: params,
            },
            'command-param-name': {
              pattern: /^(\[[\t ]*)\w+/,
              lookbehind: true,
              alias: 'name',
            },
            'start-stop-char': /[\[\]]/,
          },
        },
      },
    },
  };
  Prism.languages.nani = Prism.languages['naniscript'];

  /** @typedef {InstanceType<import("./prism-core")["Token"]>} Token */

  /**
   * This hook is used to validate generic-text tokens for balanced brackets.
   * Mark token as bad-line when contains not balanced brackets: {},[]
   */
  Prism.hooks.add('after-tokenize', function (env) {
    /** @type {(Token | string)[]} */
    var tokens = env.tokens;
    tokens.forEach(function (token) {
      if (typeof token !== 'string' && token.type === 'generic-text') {
        var content = getTextContent(token);
        if (!isBracketsBalanced(content)) {
          token.type = 'bad-line';
          token.content = content;
        }
      }
    });
  });

  /**
   * @param {string} input
   * @returns {boolean}
   */
  function isBracketsBalanced(input) {
    var brackets = '[]{}';
    var stack = [];
    for (var i = 0; i < input.length; i++) {
      var bracket = input[i];
      var bracketsIndex = brackets.indexOf(bracket);
      if (bracketsIndex !== -1) {
        if (bracketsIndex % 2 === 0) {
          stack.push(bracketsIndex + 1);
        } else if (stack.pop() !== bracketsIndex) {
          return false;
        }
      }
    }
    return stack.length === 0;
  }

  /**
   * @param {string | Token | (string | Token)[]} token
   * @returns {string}
   */
  function getTextContent(token) {
    if (typeof token === 'string') {
      return token;
    } else if (Array.isArray(token)) {
      return token.map(getTextContent).join('');
    } else {
      return getTextContent(token.content);
    }
  }
})(Prism);