You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							195 lines
						
					
					
						
							5.9 KiB
						
					
					
				
			
		
		
	
	
							195 lines
						
					
					
						
							5.9 KiB
						
					
					
				| // 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("ebnf", function (config) {
 | |
|     var commentType = {slash: 0, parenthesis: 1};
 | |
|     var stateType = {comment: 0, _string: 1, characterClass: 2};
 | |
|     var bracesMode = null;
 | |
| 
 | |
|     if (config.bracesMode)
 | |
|       bracesMode = CodeMirror.getMode(config, config.bracesMode);
 | |
| 
 | |
|     return {
 | |
|       startState: function () {
 | |
|         return {
 | |
|           stringType: null,
 | |
|           commentType: null,
 | |
|           braced: 0,
 | |
|           lhs: true,
 | |
|           localState: null,
 | |
|           stack: [],
 | |
|           inDefinition: false
 | |
|         };
 | |
|       },
 | |
|       token: function (stream, state) {
 | |
|         if (!stream) return;
 | |
| 
 | |
|         //check for state changes
 | |
|         if (state.stack.length === 0) {
 | |
|           //strings
 | |
|           if ((stream.peek() == '"') || (stream.peek() == "'")) {
 | |
|             state.stringType = stream.peek();
 | |
|             stream.next(); // Skip quote
 | |
|             state.stack.unshift(stateType._string);
 | |
|           } else if (stream.match(/^\/\*/)) { //comments starting with /*
 | |
|             state.stack.unshift(stateType.comment);
 | |
|             state.commentType = commentType.slash;
 | |
|           } else if (stream.match(/^\(\*/)) { //comments starting with (*
 | |
|             state.stack.unshift(stateType.comment);
 | |
|             state.commentType = commentType.parenthesis;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         //return state
 | |
|         //stack has
 | |
|         switch (state.stack[0]) {
 | |
|         case stateType._string:
 | |
|           while (state.stack[0] === stateType._string && !stream.eol()) {
 | |
|             if (stream.peek() === state.stringType) {
 | |
|               stream.next(); // Skip quote
 | |
|               state.stack.shift(); // Clear flag
 | |
|             } else if (stream.peek() === "\\") {
 | |
|               stream.next();
 | |
|               stream.next();
 | |
|             } else {
 | |
|               stream.match(/^.[^\\\"\']*/);
 | |
|             }
 | |
|           }
 | |
|           return state.lhs ? "property string" : "string"; // Token style
 | |
| 
 | |
|         case stateType.comment:
 | |
|           while (state.stack[0] === stateType.comment && !stream.eol()) {
 | |
|             if (state.commentType === commentType.slash && stream.match(/\*\//)) {
 | |
|               state.stack.shift(); // Clear flag
 | |
|               state.commentType = null;
 | |
|             } else if (state.commentType === commentType.parenthesis && stream.match(/\*\)/)) {
 | |
|               state.stack.shift(); // Clear flag
 | |
|               state.commentType = null;
 | |
|             } else {
 | |
|               stream.match(/^.[^\*]*/);
 | |
|             }
 | |
|           }
 | |
|           return "comment";
 | |
| 
 | |
|         case stateType.characterClass:
 | |
|           while (state.stack[0] === stateType.characterClass && !stream.eol()) {
 | |
|             if (!(stream.match(/^[^\]\\]+/) || stream.match(/^\\./))) {
 | |
|               state.stack.shift();
 | |
|             }
 | |
|           }
 | |
|           return "operator";
 | |
|         }
 | |
| 
 | |
|         var peek = stream.peek();
 | |
| 
 | |
|         if (bracesMode !== null && (state.braced || peek === "{")) {
 | |
|           if (state.localState === null)
 | |
|             state.localState = CodeMirror.startState(bracesMode);
 | |
| 
 | |
|           var token = bracesMode.token(stream, state.localState),
 | |
|           text = stream.current();
 | |
| 
 | |
|           if (!token) {
 | |
|             for (var i = 0; i < text.length; i++) {
 | |
|               if (text[i] === "{") {
 | |
|                 if (state.braced === 0) {
 | |
|                   token = "matchingbracket";
 | |
|                 }
 | |
|                 state.braced++;
 | |
|               } else if (text[i] === "}") {
 | |
|                 state.braced--;
 | |
|                 if (state.braced === 0) {
 | |
|                   token = "matchingbracket";
 | |
|                 }
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|           return token;
 | |
|         }
 | |
| 
 | |
|         //no stack
 | |
|         switch (peek) {
 | |
|         case "[":
 | |
|           stream.next();
 | |
|           state.stack.unshift(stateType.characterClass);
 | |
|           return "bracket";
 | |
|         case ":":
 | |
|         case "|":
 | |
|         case ";":
 | |
|           stream.next();
 | |
|           return "operator";
 | |
|         case "%":
 | |
|           if (stream.match("%%")) {
 | |
|             return "header";
 | |
|           } else if (stream.match(/[%][A-Za-z]+/)) {
 | |
|             return "keyword";
 | |
|           } else if (stream.match(/[%][}]/)) {
 | |
|             return "matchingbracket";
 | |
|           }
 | |
|           break;
 | |
|         case "/":
 | |
|           if (stream.match(/[\/][A-Za-z]+/)) {
 | |
|           return "keyword";
 | |
|         }
 | |
|         case "\\":
 | |
|           if (stream.match(/[\][a-z]+/)) {
 | |
|             return "string-2";
 | |
|           }
 | |
|         case ".":
 | |
|           if (stream.match(".")) {
 | |
|             return "atom";
 | |
|           }
 | |
|         case "*":
 | |
|         case "-":
 | |
|         case "+":
 | |
|         case "^":
 | |
|           if (stream.match(peek)) {
 | |
|             return "atom";
 | |
|           }
 | |
|         case "$":
 | |
|           if (stream.match("$$")) {
 | |
|             return "builtin";
 | |
|           } else if (stream.match(/[$][0-9]+/)) {
 | |
|             return "variable-3";
 | |
|           }
 | |
|         case "<":
 | |
|           if (stream.match(/<<[a-zA-Z_]+>>/)) {
 | |
|             return "builtin";
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         if (stream.match(/^\/\//)) {
 | |
|           stream.skipToEnd();
 | |
|           return "comment";
 | |
|         } else if (stream.match(/return/)) {
 | |
|           return "operator";
 | |
|         } else if (stream.match(/^[a-zA-Z_][a-zA-Z0-9_]*/)) {
 | |
|           if (stream.match(/(?=[\(.])/)) {
 | |
|             return "variable";
 | |
|           } else if (stream.match(/(?=[\s\n]*[:=])/)) {
 | |
|             return "def";
 | |
|           }
 | |
|           return "variable-2";
 | |
|         } else if (["[", "]", "(", ")"].indexOf(stream.peek()) != -1) {
 | |
|           stream.next();
 | |
|           return "bracket";
 | |
|         } else if (!stream.eatSpace()) {
 | |
|           stream.next();
 | |
|         }
 | |
|         return null;
 | |
|       }
 | |
|     };
 | |
|   });
 | |
| 
 | |
|   CodeMirror.defineMIME("text/x-ebnf", "ebnf");
 | |
| });
 | |
| 
 |