Blame view

backend/makest/js/plugins/codemirror/mode/octave/octave.js 4.36 KB
d1f8bd40   Alexey Boroda   first commit
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
  // CodeMirror, copyright (c) by Marijn Haverbeke and others
  // Distributed under an MIT license: http://codemirror.net/LICENSE
  
  (function(mod) {
    if (typeof exports == "object" && typeof module == "object") // CommonJS
      mod(require("../../lib/codemirror"));
    else if (typeof define == "function" && define.amd) // AMD
      define(["../../lib/codemirror"], mod);
    else // Plain browser env
      mod(CodeMirror);
  })(function(CodeMirror) {
  "use strict";
  
  CodeMirror.defineMode("octave", function() {
    function wordRegexp(words) {
      return new RegExp("^((" + words.join(")|(") + "))\\b");
    }
  
    var singleOperators = new RegExp("^[\\+\\-\\*/&|\\^~<>!@'\\\\]");
    var singleDelimiters = new RegExp('^[\\(\\[\\{\\},:=;]');
    var doubleOperators = new RegExp("^((==)|(~=)|(<=)|(>=)|(<<)|(>>)|(\\.[\\+\\-\\*/\\^\\\\]))");
    var doubleDelimiters = new RegExp("^((!=)|(\\+=)|(\\-=)|(\\*=)|(/=)|(&=)|(\\|=)|(\\^=))");
    var tripleDelimiters = new RegExp("^((>>=)|(<<=))");
    var expressionEnd = new RegExp("^[\\]\\)]");
    var identifiers = new RegExp("^[_A-Za-z\xa1-\uffff][_A-Za-z0-9\xa1-\uffff]*");
  
    var builtins = wordRegexp([
      'error', 'eval', 'function', 'abs', 'acos', 'atan', 'asin', 'cos',
      'cosh', 'exp', 'log', 'prod', 'sum', 'log10', 'max', 'min', 'sign', 'sin', 'sinh',
      'sqrt', 'tan', 'reshape', 'break', 'zeros', 'default', 'margin', 'round', 'ones',
      'rand', 'syn', 'ceil', 'floor', 'size', 'clear', 'zeros', 'eye', 'mean', 'std', 'cov',
      'det', 'eig', 'inv', 'norm', 'rank', 'trace', 'expm', 'logm', 'sqrtm', 'linspace', 'plot',
      'title', 'xlabel', 'ylabel', 'legend', 'text', 'grid', 'meshgrid', 'mesh', 'num2str',
      'fft', 'ifft', 'arrayfun', 'cellfun', 'input', 'fliplr', 'flipud', 'ismember'
    ]);
  
    var keywords = wordRegexp([
      'return', 'case', 'switch', 'else', 'elseif', 'end', 'endif', 'endfunction',
      'if', 'otherwise', 'do', 'for', 'while', 'try', 'catch', 'classdef', 'properties', 'events',
      'methods', 'global', 'persistent', 'endfor', 'endwhile', 'printf', 'sprintf', 'disp', 'until',
      'continue', 'pkg'
    ]);
  
  
    // tokenizers
    function tokenTranspose(stream, state) {
      if (!stream.sol() && stream.peek() === '\'') {
        stream.next();
        state.tokenize = tokenBase;
        return 'operator';
      }
      state.tokenize = tokenBase;
      return tokenBase(stream, state);
    }
  
  
    function tokenComment(stream, state) {
      if (stream.match(/^.*%}/)) {
        state.tokenize = tokenBase;
        return 'comment';
      };
      stream.skipToEnd();
      return 'comment';
    }
  
    function tokenBase(stream, state) {
      // whitespaces
      if (stream.eatSpace()) return null;
  
      // Handle one line Comments
      if (stream.match('%{')){
        state.tokenize = tokenComment;
        stream.skipToEnd();
        return 'comment';
      }
  
      if (stream.match(/^[%#]/)){
        stream.skipToEnd();
        return 'comment';
      }
  
      // Handle Number Literals
      if (stream.match(/^[0-9\.+-]/, false)) {
        if (stream.match(/^[+-]?0x[0-9a-fA-F]+[ij]?/)) {
          stream.tokenize = tokenBase;
          return 'number'; };
        if (stream.match(/^[+-]?\d*\.\d+([EeDd][+-]?\d+)?[ij]?/)) { return 'number'; };
        if (stream.match(/^[+-]?\d+([EeDd][+-]?\d+)?[ij]?/)) { return 'number'; };
      }
      if (stream.match(wordRegexp(['nan','NaN','inf','Inf']))) { return 'number'; };
  
      // Handle Strings
      if (stream.match(/^"([^"]|(""))*"/)) { return 'string'; } ;
      if (stream.match(/^'([^']|(''))*'/)) { return 'string'; } ;
  
      // Handle words
      if (stream.match(keywords)) { return 'keyword'; } ;
      if (stream.match(builtins)) { return 'builtin'; } ;
      if (stream.match(identifiers)) { return 'variable'; } ;
  
      if (stream.match(singleOperators) || stream.match(doubleOperators)) { return 'operator'; };
      if (stream.match(singleDelimiters) || stream.match(doubleDelimiters) || stream.match(tripleDelimiters)) { return null; };
  
      if (stream.match(expressionEnd)) {
        state.tokenize = tokenTranspose;
        return null;
      };
  
  
      // Handle non-detected items
      stream.next();
      return 'error';
    };
  
  
    return {
      startState: function() {
        return {
          tokenize: tokenBase
        };
      },
  
      token: function(stream, state) {
        var style = state.tokenize(stream, state);
        if (style === 'number' || style === 'variable'){
          state.tokenize = tokenTranspose;
        }
        return style;
      }
    };
  });
  
  CodeMirror.defineMIME("text/x-octave", "octave");
  
  });