dew/src/main/resources/org/apidesign/bck2brwsr/dew/js/codemirror/mode/xml.js
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 23 Jan 2013 13:18:46 +0100
branchdew
changeset 544 08ffdc3938e7
parent 460 launcher/src/main/resources/org/apidesign/bck2brwsr/dew/js/codemirror/mode/xml.js@c0f1788183dd
permissions -rw-r--r--
Moving Development Environment for Web to own project
phrebejk@460
     1
CodeMirror.defineMode("xml", function(config, parserConfig) {
phrebejk@460
     2
  var indentUnit = config.indentUnit;
phrebejk@460
     3
  var Kludges = parserConfig.htmlMode ? {
phrebejk@460
     4
    autoSelfClosers: {'area': true, 'base': true, 'br': true, 'col': true, 'command': true,
phrebejk@460
     5
                      'embed': true, 'frame': true, 'hr': true, 'img': true, 'input': true,
phrebejk@460
     6
                      'keygen': true, 'link': true, 'meta': true, 'param': true, 'source': true,
phrebejk@460
     7
                      'track': true, 'wbr': true},
phrebejk@460
     8
    implicitlyClosed: {'dd': true, 'li': true, 'optgroup': true, 'option': true, 'p': true,
phrebejk@460
     9
                       'rp': true, 'rt': true, 'tbody': true, 'td': true, 'tfoot': true,
phrebejk@460
    10
                       'th': true, 'tr': true},
phrebejk@460
    11
    contextGrabbers: {
phrebejk@460
    12
      'dd': {'dd': true, 'dt': true},
phrebejk@460
    13
      'dt': {'dd': true, 'dt': true},
phrebejk@460
    14
      'li': {'li': true},
phrebejk@460
    15
      'option': {'option': true, 'optgroup': true},
phrebejk@460
    16
      'optgroup': {'optgroup': true},
phrebejk@460
    17
      'p': {'address': true, 'article': true, 'aside': true, 'blockquote': true, 'dir': true,
phrebejk@460
    18
            'div': true, 'dl': true, 'fieldset': true, 'footer': true, 'form': true,
phrebejk@460
    19
            'h1': true, 'h2': true, 'h3': true, 'h4': true, 'h5': true, 'h6': true,
phrebejk@460
    20
            'header': true, 'hgroup': true, 'hr': true, 'menu': true, 'nav': true, 'ol': true,
phrebejk@460
    21
            'p': true, 'pre': true, 'section': true, 'table': true, 'ul': true},
phrebejk@460
    22
      'rp': {'rp': true, 'rt': true},
phrebejk@460
    23
      'rt': {'rp': true, 'rt': true},
phrebejk@460
    24
      'tbody': {'tbody': true, 'tfoot': true},
phrebejk@460
    25
      'td': {'td': true, 'th': true},
phrebejk@460
    26
      'tfoot': {'tbody': true},
phrebejk@460
    27
      'th': {'td': true, 'th': true},
phrebejk@460
    28
      'thead': {'tbody': true, 'tfoot': true},
phrebejk@460
    29
      'tr': {'tr': true}
phrebejk@460
    30
    },
phrebejk@460
    31
    doNotIndent: {"pre": true},
phrebejk@460
    32
    allowUnquoted: true,
phrebejk@460
    33
    allowMissing: true
phrebejk@460
    34
  } : {
phrebejk@460
    35
    autoSelfClosers: {},
phrebejk@460
    36
    implicitlyClosed: {},
phrebejk@460
    37
    contextGrabbers: {},
phrebejk@460
    38
    doNotIndent: {},
phrebejk@460
    39
    allowUnquoted: false,
phrebejk@460
    40
    allowMissing: false
phrebejk@460
    41
  };
phrebejk@460
    42
  var alignCDATA = parserConfig.alignCDATA;
phrebejk@460
    43
phrebejk@460
    44
  // Return variables for tokenizers
phrebejk@460
    45
  var tagName, type;
phrebejk@460
    46
phrebejk@460
    47
  function inText(stream, state) {
phrebejk@460
    48
    function chain(parser) {
phrebejk@460
    49
      state.tokenize = parser;
phrebejk@460
    50
      return parser(stream, state);
phrebejk@460
    51
    }
phrebejk@460
    52
phrebejk@460
    53
    var ch = stream.next();
phrebejk@460
    54
    if (ch == "<") {
phrebejk@460
    55
      if (stream.eat("!")) {
phrebejk@460
    56
        if (stream.eat("[")) {
phrebejk@460
    57
          if (stream.match("CDATA[")) return chain(inBlock("atom", "]]>"));
phrebejk@460
    58
          else return null;
phrebejk@460
    59
        }
phrebejk@460
    60
        else if (stream.match("--")) return chain(inBlock("comment", "-->"));
phrebejk@460
    61
        else if (stream.match("DOCTYPE", true, true)) {
phrebejk@460
    62
          stream.eatWhile(/[\w\._\-]/);
phrebejk@460
    63
          return chain(doctype(1));
phrebejk@460
    64
        }
phrebejk@460
    65
        else return null;
phrebejk@460
    66
      }
phrebejk@460
    67
      else if (stream.eat("?")) {
phrebejk@460
    68
        stream.eatWhile(/[\w\._\-]/);
phrebejk@460
    69
        state.tokenize = inBlock("meta", "?>");
phrebejk@460
    70
        return "meta";
phrebejk@460
    71
      }
phrebejk@460
    72
      else {
phrebejk@460
    73
        var isClose = stream.eat("/");
phrebejk@460
    74
        tagName = "";
phrebejk@460
    75
        var c;
phrebejk@460
    76
        while ((c = stream.eat(/[^\s\u00a0=<>\"\'\/?]/))) tagName += c;
phrebejk@460
    77
        if (!tagName) return "error";
phrebejk@460
    78
        type = isClose ? "closeTag" : "openTag";
phrebejk@460
    79
        state.tokenize = inTag;
phrebejk@460
    80
        return "tag";
phrebejk@460
    81
      }
phrebejk@460
    82
    }
phrebejk@460
    83
    else if (ch == "&") {
phrebejk@460
    84
      var ok;
phrebejk@460
    85
      if (stream.eat("#")) {
phrebejk@460
    86
        if (stream.eat("x")) {
phrebejk@460
    87
          ok = stream.eatWhile(/[a-fA-F\d]/) && stream.eat(";");          
phrebejk@460
    88
        } else {
phrebejk@460
    89
          ok = stream.eatWhile(/[\d]/) && stream.eat(";");
phrebejk@460
    90
        }
phrebejk@460
    91
      } else {
phrebejk@460
    92
        ok = stream.eatWhile(/[\w\.\-:]/) && stream.eat(";");
phrebejk@460
    93
      }
phrebejk@460
    94
      return ok ? "atom" : "error";
phrebejk@460
    95
    }
phrebejk@460
    96
    else {
phrebejk@460
    97
      stream.eatWhile(/[^&<]/);
phrebejk@460
    98
      return null;
phrebejk@460
    99
    }
phrebejk@460
   100
  }
phrebejk@460
   101
phrebejk@460
   102
  function inTag(stream, state) {
phrebejk@460
   103
    var ch = stream.next();
phrebejk@460
   104
    if (ch == ">" || (ch == "/" && stream.eat(">"))) {
phrebejk@460
   105
      state.tokenize = inText;
phrebejk@460
   106
      type = ch == ">" ? "endTag" : "selfcloseTag";
phrebejk@460
   107
      return "tag";
phrebejk@460
   108
    }
phrebejk@460
   109
    else if (ch == "=") {
phrebejk@460
   110
      type = "equals";
phrebejk@460
   111
      return null;
phrebejk@460
   112
    }
phrebejk@460
   113
    else if (/[\'\"]/.test(ch)) {
phrebejk@460
   114
      state.tokenize = inAttribute(ch);
phrebejk@460
   115
      return state.tokenize(stream, state);
phrebejk@460
   116
    }
phrebejk@460
   117
    else {
phrebejk@460
   118
      stream.eatWhile(/[^\s\u00a0=<>\"\']/);
phrebejk@460
   119
      return "word";
phrebejk@460
   120
    }
phrebejk@460
   121
  }
phrebejk@460
   122
phrebejk@460
   123
  function inAttribute(quote) {
phrebejk@460
   124
    return function(stream, state) {
phrebejk@460
   125
      while (!stream.eol()) {
phrebejk@460
   126
        if (stream.next() == quote) {
phrebejk@460
   127
          state.tokenize = inTag;
phrebejk@460
   128
          break;
phrebejk@460
   129
        }
phrebejk@460
   130
      }
phrebejk@460
   131
      return "string";
phrebejk@460
   132
    };
phrebejk@460
   133
  }
phrebejk@460
   134
phrebejk@460
   135
  function inBlock(style, terminator) {
phrebejk@460
   136
    return function(stream, state) {
phrebejk@460
   137
      while (!stream.eol()) {
phrebejk@460
   138
        if (stream.match(terminator)) {
phrebejk@460
   139
          state.tokenize = inText;
phrebejk@460
   140
          break;
phrebejk@460
   141
        }
phrebejk@460
   142
        stream.next();
phrebejk@460
   143
      }
phrebejk@460
   144
      return style;
phrebejk@460
   145
    };
phrebejk@460
   146
  }
phrebejk@460
   147
  function doctype(depth) {
phrebejk@460
   148
    return function(stream, state) {
phrebejk@460
   149
      var ch;
phrebejk@460
   150
      while ((ch = stream.next()) != null) {
phrebejk@460
   151
        if (ch == "<") {
phrebejk@460
   152
          state.tokenize = doctype(depth + 1);
phrebejk@460
   153
          return state.tokenize(stream, state);
phrebejk@460
   154
        } else if (ch == ">") {
phrebejk@460
   155
          if (depth == 1) {
phrebejk@460
   156
            state.tokenize = inText;
phrebejk@460
   157
            break;
phrebejk@460
   158
          } else {
phrebejk@460
   159
            state.tokenize = doctype(depth - 1);
phrebejk@460
   160
            return state.tokenize(stream, state);
phrebejk@460
   161
          }
phrebejk@460
   162
        }
phrebejk@460
   163
      }
phrebejk@460
   164
      return "meta";
phrebejk@460
   165
    };
phrebejk@460
   166
  }
phrebejk@460
   167
phrebejk@460
   168
  var curState, setStyle;
phrebejk@460
   169
  function pass() {
phrebejk@460
   170
    for (var i = arguments.length - 1; i >= 0; i--) curState.cc.push(arguments[i]);
phrebejk@460
   171
  }
phrebejk@460
   172
  function cont() {
phrebejk@460
   173
    pass.apply(null, arguments);
phrebejk@460
   174
    return true;
phrebejk@460
   175
  }
phrebejk@460
   176
phrebejk@460
   177
  function pushContext(tagName, startOfLine) {
phrebejk@460
   178
    var noIndent = Kludges.doNotIndent.hasOwnProperty(tagName) || (curState.context && curState.context.noIndent);
phrebejk@460
   179
    curState.context = {
phrebejk@460
   180
      prev: curState.context,
phrebejk@460
   181
      tagName: tagName,
phrebejk@460
   182
      indent: curState.indented,
phrebejk@460
   183
      startOfLine: startOfLine,
phrebejk@460
   184
      noIndent: noIndent
phrebejk@460
   185
    };
phrebejk@460
   186
  }
phrebejk@460
   187
  function popContext() {
phrebejk@460
   188
    if (curState.context) curState.context = curState.context.prev;
phrebejk@460
   189
  }
phrebejk@460
   190
phrebejk@460
   191
  function element(type) {
phrebejk@460
   192
    if (type == "openTag") {
phrebejk@460
   193
      curState.tagName = tagName;
phrebejk@460
   194
      return cont(attributes, endtag(curState.startOfLine));
phrebejk@460
   195
    } else if (type == "closeTag") {
phrebejk@460
   196
      var err = false;
phrebejk@460
   197
      if (curState.context) {
phrebejk@460
   198
        if (curState.context.tagName != tagName) {
phrebejk@460
   199
          if (Kludges.implicitlyClosed.hasOwnProperty(curState.context.tagName.toLowerCase())) {
phrebejk@460
   200
            popContext();
phrebejk@460
   201
          }
phrebejk@460
   202
          err = !curState.context || curState.context.tagName != tagName;
phrebejk@460
   203
        }
phrebejk@460
   204
      } else {
phrebejk@460
   205
        err = true;
phrebejk@460
   206
      }
phrebejk@460
   207
      if (err) setStyle = "error";
phrebejk@460
   208
      return cont(endclosetag(err));
phrebejk@460
   209
    }
phrebejk@460
   210
    return cont();
phrebejk@460
   211
  }
phrebejk@460
   212
  function endtag(startOfLine) {
phrebejk@460
   213
    return function(type) {
phrebejk@460
   214
      var tagName = curState.tagName;
phrebejk@460
   215
      curState.tagName = null;
phrebejk@460
   216
      if (type == "selfcloseTag" ||
phrebejk@460
   217
          (type == "endTag" && Kludges.autoSelfClosers.hasOwnProperty(tagName.toLowerCase()))) {
phrebejk@460
   218
        maybePopContext(tagName.toLowerCase());
phrebejk@460
   219
        return cont();
phrebejk@460
   220
      }
phrebejk@460
   221
      if (type == "endTag") {
phrebejk@460
   222
        maybePopContext(tagName.toLowerCase());
phrebejk@460
   223
        pushContext(tagName, startOfLine);
phrebejk@460
   224
        return cont();
phrebejk@460
   225
      }
phrebejk@460
   226
      return cont();
phrebejk@460
   227
    };
phrebejk@460
   228
  }
phrebejk@460
   229
  function endclosetag(err) {
phrebejk@460
   230
    return function(type) {
phrebejk@460
   231
      if (err) setStyle = "error";
phrebejk@460
   232
      if (type == "endTag") { popContext(); return cont(); }
phrebejk@460
   233
      setStyle = "error";
phrebejk@460
   234
      return cont(arguments.callee);
phrebejk@460
   235
    };
phrebejk@460
   236
  }
phrebejk@460
   237
  function maybePopContext(nextTagName) {
phrebejk@460
   238
    var parentTagName;
phrebejk@460
   239
    while (true) {
phrebejk@460
   240
      if (!curState.context) {
phrebejk@460
   241
        return;
phrebejk@460
   242
      }
phrebejk@460
   243
      parentTagName = curState.context.tagName.toLowerCase();
phrebejk@460
   244
      if (!Kludges.contextGrabbers.hasOwnProperty(parentTagName) ||
phrebejk@460
   245
          !Kludges.contextGrabbers[parentTagName].hasOwnProperty(nextTagName)) {
phrebejk@460
   246
        return;
phrebejk@460
   247
      }
phrebejk@460
   248
      popContext();
phrebejk@460
   249
    }
phrebejk@460
   250
  }
phrebejk@460
   251
phrebejk@460
   252
  function attributes(type) {
phrebejk@460
   253
    if (type == "word") {setStyle = "attribute"; return cont(attribute, attributes);}
phrebejk@460
   254
    if (type == "endTag" || type == "selfcloseTag") return pass();
phrebejk@460
   255
    setStyle = "error";
phrebejk@460
   256
    return cont(attributes);
phrebejk@460
   257
  }
phrebejk@460
   258
  function attribute(type) {
phrebejk@460
   259
    if (type == "equals") return cont(attvalue, attributes);
phrebejk@460
   260
    if (!Kludges.allowMissing) setStyle = "error";
phrebejk@460
   261
    else if (type == "word") setStyle = "attribute";
phrebejk@460
   262
    return (type == "endTag" || type == "selfcloseTag") ? pass() : cont();
phrebejk@460
   263
  }
phrebejk@460
   264
  function attvalue(type) {
phrebejk@460
   265
    if (type == "string") return cont(attvaluemaybe);
phrebejk@460
   266
    if (type == "word" && Kludges.allowUnquoted) {setStyle = "string"; return cont();}
phrebejk@460
   267
    setStyle = "error";
phrebejk@460
   268
    return (type == "endTag" || type == "selfCloseTag") ? pass() : cont();
phrebejk@460
   269
  }
phrebejk@460
   270
  function attvaluemaybe(type) {
phrebejk@460
   271
    if (type == "string") return cont(attvaluemaybe);
phrebejk@460
   272
    else return pass();
phrebejk@460
   273
  }
phrebejk@460
   274
phrebejk@460
   275
  return {
phrebejk@460
   276
    startState: function() {
phrebejk@460
   277
      return {tokenize: inText, cc: [], indented: 0, startOfLine: true, tagName: null, context: null};
phrebejk@460
   278
    },
phrebejk@460
   279
phrebejk@460
   280
    token: function(stream, state) {
phrebejk@460
   281
      if (stream.sol()) {
phrebejk@460
   282
        state.startOfLine = true;
phrebejk@460
   283
        state.indented = stream.indentation();
phrebejk@460
   284
      }
phrebejk@460
   285
      if (stream.eatSpace()) return null;
phrebejk@460
   286
phrebejk@460
   287
      setStyle = type = tagName = null;
phrebejk@460
   288
      var style = state.tokenize(stream, state);
phrebejk@460
   289
      state.type = type;
phrebejk@460
   290
      if ((style || type) && style != "comment") {
phrebejk@460
   291
        curState = state;
phrebejk@460
   292
        while (true) {
phrebejk@460
   293
          var comb = state.cc.pop() || element;
phrebejk@460
   294
          if (comb(type || style)) break;
phrebejk@460
   295
        }
phrebejk@460
   296
      }
phrebejk@460
   297
      state.startOfLine = false;
phrebejk@460
   298
      return setStyle || style;
phrebejk@460
   299
    },
phrebejk@460
   300
phrebejk@460
   301
    indent: function(state, textAfter, fullLine) {
phrebejk@460
   302
      var context = state.context;
phrebejk@460
   303
      if ((state.tokenize != inTag && state.tokenize != inText) ||
phrebejk@460
   304
          context && context.noIndent)
phrebejk@460
   305
        return fullLine ? fullLine.match(/^(\s*)/)[0].length : 0;
phrebejk@460
   306
      if (alignCDATA && /<!\[CDATA\[/.test(textAfter)) return 0;
phrebejk@460
   307
      if (context && /^<\//.test(textAfter))
phrebejk@460
   308
        context = context.prev;
phrebejk@460
   309
      while (context && !context.startOfLine)
phrebejk@460
   310
        context = context.prev;
phrebejk@460
   311
      if (context) return context.indent + indentUnit;
phrebejk@460
   312
      else return 0;
phrebejk@460
   313
    },
phrebejk@460
   314
phrebejk@460
   315
    electricChars: "/",
phrebejk@460
   316
phrebejk@460
   317
    configuration: parserConfig.htmlMode ? "html" : "xml"
phrebejk@460
   318
  };
phrebejk@460
   319
});
phrebejk@460
   320
phrebejk@460
   321
CodeMirror.defineMIME("text/xml", "xml");
phrebejk@460
   322
CodeMirror.defineMIME("application/xml", "xml");
phrebejk@460
   323
if (!CodeMirror.mimeModes.hasOwnProperty("text/html"))
phrebejk@460
   324
  CodeMirror.defineMIME("text/html", {name: "xml", htmlMode: true});