| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  | // CodeMirror, copyright (c) by Marijn Haverbeke and others
 | 
					
						
							| 
									
										
										
										
											2018-10-07 12:02:07 +02:00
										 |  |  | // Distributed under an MIT license: https://codemirror.net/LICENSE
 | 
					
						
							| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Stylus mode created by Dmitry Kiselyov http://git.io/AaRB
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | (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("stylus", function(config) { | 
					
						
							|  |  |  |     var indentUnit = config.indentUnit, | 
					
						
							|  |  |  |         indentUnitString = '', | 
					
						
							|  |  |  |         tagKeywords = keySet(tagKeywords_), | 
					
						
							|  |  |  |         tagVariablesRegexp = /^(a|b|i|s|col|em)$/i, | 
					
						
							|  |  |  |         propertyKeywords = keySet(propertyKeywords_), | 
					
						
							|  |  |  |         nonStandardPropertyKeywords = keySet(nonStandardPropertyKeywords_), | 
					
						
							|  |  |  |         valueKeywords = keySet(valueKeywords_), | 
					
						
							|  |  |  |         colorKeywords = keySet(colorKeywords_), | 
					
						
							|  |  |  |         documentTypes = keySet(documentTypes_), | 
					
						
							|  |  |  |         documentTypesRegexp = wordRegexp(documentTypes_), | 
					
						
							|  |  |  |         mediaFeatures = keySet(mediaFeatures_), | 
					
						
							|  |  |  |         mediaTypes = keySet(mediaTypes_), | 
					
						
							|  |  |  |         fontProperties = keySet(fontProperties_), | 
					
						
							|  |  |  |         operatorsRegexp = /^\s*([.]{2,3}|&&|\|\||\*\*|[?!=:]?=|[-+*\/%<>]=?|\?:|\~)/, | 
					
						
							|  |  |  |         wordOperatorKeywordsRegexp = wordRegexp(wordOperatorKeywords_), | 
					
						
							|  |  |  |         blockKeywords = keySet(blockKeywords_), | 
					
						
							|  |  |  |         vendorPrefixesRegexp = new RegExp(/^\-(moz|ms|o|webkit)-/i), | 
					
						
							|  |  |  |         commonAtoms = keySet(commonAtoms_), | 
					
						
							|  |  |  |         firstWordMatch = "", | 
					
						
							|  |  |  |         states = {}, | 
					
						
							|  |  |  |         ch, | 
					
						
							|  |  |  |         style, | 
					
						
							|  |  |  |         type, | 
					
						
							|  |  |  |         override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (indentUnitString.length < indentUnit) indentUnitString += ' '; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Tokenizers | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function tokenBase(stream, state) { | 
					
						
							|  |  |  |       firstWordMatch = stream.string.match(/(^[\w-]+\s*=\s*$)|(^\s*[\w-]+\s*=\s*[\w-])|(^\s*(\.|#|@|\$|\&|\[|\d|\+|::?|\{|\>|~|\/)?\s*[\w-]*([a-z0-9-]|\*|\/\*)(\(|,)?)/); | 
					
						
							|  |  |  |       state.context.line.firstWord = firstWordMatch ? firstWordMatch[0].replace(/^\s*/, "") : ""; | 
					
						
							|  |  |  |       state.context.line.indent = stream.indentation(); | 
					
						
							|  |  |  |       ch = stream.peek(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Line comment
 | 
					
						
							|  |  |  |       if (stream.match("//")) { | 
					
						
							|  |  |  |         stream.skipToEnd(); | 
					
						
							|  |  |  |         return ["comment", "comment"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Block comment
 | 
					
						
							|  |  |  |       if (stream.match("/*")) { | 
					
						
							|  |  |  |         state.tokenize = tokenCComment; | 
					
						
							|  |  |  |         return tokenCComment(stream, state); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // String
 | 
					
						
							|  |  |  |       if (ch == "\"" || ch == "'") { | 
					
						
							|  |  |  |         stream.next(); | 
					
						
							|  |  |  |         state.tokenize = tokenString(ch); | 
					
						
							|  |  |  |         return state.tokenize(stream, state); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Def
 | 
					
						
							|  |  |  |       if (ch == "@") { | 
					
						
							|  |  |  |         stream.next(); | 
					
						
							|  |  |  |         stream.eatWhile(/[\w\\-]/); | 
					
						
							|  |  |  |         return ["def", stream.current()]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // ID selector or Hex color
 | 
					
						
							|  |  |  |       if (ch == "#") { | 
					
						
							|  |  |  |         stream.next(); | 
					
						
							|  |  |  |         // Hex color
 | 
					
						
							| 
									
										
										
										
											2020-09-20 21:41:40 +02:00
										 |  |  |         if (stream.match(/^[0-9a-f]{3}([0-9a-f]([0-9a-f]{2}){0,2})?\b(?!-)/i)) { | 
					
						
							| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  |           return ["atom", "atom"]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // ID selector
 | 
					
						
							|  |  |  |         if (stream.match(/^[a-z][\w-]*/i)) { | 
					
						
							|  |  |  |           return ["builtin", "hash"]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Vendor prefixes
 | 
					
						
							|  |  |  |       if (stream.match(vendorPrefixesRegexp)) { | 
					
						
							|  |  |  |         return ["meta", "vendor-prefixes"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Numbers
 | 
					
						
							|  |  |  |       if (stream.match(/^-?[0-9]?\.?[0-9]/)) { | 
					
						
							|  |  |  |         stream.eatWhile(/[a-z%]/i); | 
					
						
							|  |  |  |         return ["number", "unit"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // !important|optional
 | 
					
						
							|  |  |  |       if (ch == "!") { | 
					
						
							|  |  |  |         stream.next(); | 
					
						
							|  |  |  |         return [stream.match(/^(important|optional)/i) ? "keyword": "operator", "important"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Class
 | 
					
						
							|  |  |  |       if (ch == "." && stream.match(/^\.[a-z][\w-]*/i)) { | 
					
						
							|  |  |  |         return ["qualifier", "qualifier"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // url url-prefix domain regexp
 | 
					
						
							|  |  |  |       if (stream.match(documentTypesRegexp)) { | 
					
						
							|  |  |  |         if (stream.peek() == "(") state.tokenize = tokenParenthesized; | 
					
						
							|  |  |  |         return ["property", "word"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Mixins / Functions
 | 
					
						
							|  |  |  |       if (stream.match(/^[a-z][\w-]*\(/i)) { | 
					
						
							|  |  |  |         stream.backUp(1); | 
					
						
							|  |  |  |         return ["keyword", "mixin"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Block mixins
 | 
					
						
							|  |  |  |       if (stream.match(/^(\+|-)[a-z][\w-]*\(/i)) { | 
					
						
							|  |  |  |         stream.backUp(1); | 
					
						
							|  |  |  |         return ["keyword", "block-mixin"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Parent Reference BEM naming
 | 
					
						
							|  |  |  |       if (stream.string.match(/^\s*&/) && stream.match(/^[-_]+[a-z][\w-]*/)) { | 
					
						
							|  |  |  |         return ["qualifier", "qualifier"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // / Root Reference & Parent Reference
 | 
					
						
							|  |  |  |       if (stream.match(/^(\/|&)(-|_|:|\.|#|[a-z])/)) { | 
					
						
							|  |  |  |         stream.backUp(1); | 
					
						
							|  |  |  |         return ["variable-3", "reference"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (stream.match(/^&{1}\s*$/)) { | 
					
						
							|  |  |  |         return ["variable-3", "reference"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Word operator
 | 
					
						
							|  |  |  |       if (stream.match(wordOperatorKeywordsRegexp)) { | 
					
						
							|  |  |  |         return ["operator", "operator"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Word
 | 
					
						
							|  |  |  |       if (stream.match(/^\$?[-_]*[a-z0-9]+[\w-]*/i)) { | 
					
						
							|  |  |  |         // Variable
 | 
					
						
							|  |  |  |         if (stream.match(/^(\.|\[)[\w-\'\"\]]+/i, false)) { | 
					
						
							|  |  |  |           if (!wordIsTag(stream.current())) { | 
					
						
							| 
									
										
										
										
											2021-04-06 22:16:34 +02:00
										 |  |  |             stream.match('.'); | 
					
						
							| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  |             return ["variable-2", "variable-name"]; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return ["variable-2", "word"]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Operators
 | 
					
						
							|  |  |  |       if (stream.match(operatorsRegexp)) { | 
					
						
							|  |  |  |         return ["operator", stream.current()]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Delimiters
 | 
					
						
							|  |  |  |       if (/[:;,{}\[\]\(\)]/.test(ch)) { | 
					
						
							|  |  |  |         stream.next(); | 
					
						
							|  |  |  |         return [null, ch]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // Non-detected items
 | 
					
						
							|  |  |  |       stream.next(); | 
					
						
							|  |  |  |       return [null, null]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Token comment | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function tokenCComment(stream, state) { | 
					
						
							|  |  |  |       var maybeEnd = false, ch; | 
					
						
							|  |  |  |       while ((ch = stream.next()) != null) { | 
					
						
							|  |  |  |         if (maybeEnd && ch == "/") { | 
					
						
							|  |  |  |           state.tokenize = null; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         maybeEnd = (ch == "*"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return ["comment", "comment"]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Token string | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function tokenString(quote) { | 
					
						
							|  |  |  |       return function(stream, state) { | 
					
						
							|  |  |  |         var escaped = false, ch; | 
					
						
							|  |  |  |         while ((ch = stream.next()) != null) { | 
					
						
							|  |  |  |           if (ch == quote && !escaped) { | 
					
						
							|  |  |  |             if (quote == ")") stream.backUp(1); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           escaped = !escaped && ch == "\\"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (ch == quote || !escaped && quote != ")") state.tokenize = null; | 
					
						
							|  |  |  |         return ["string", "string"]; | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Token parenthesized | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function tokenParenthesized(stream, state) { | 
					
						
							|  |  |  |       stream.next(); // Must be "("
 | 
					
						
							|  |  |  |       if (!stream.match(/\s*[\"\')]/, false)) | 
					
						
							|  |  |  |         state.tokenize = tokenString(")"); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         state.tokenize = null; | 
					
						
							|  |  |  |       return [null, "("]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Context management | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function Context(type, indent, prev, line) { | 
					
						
							|  |  |  |       this.type = type; | 
					
						
							|  |  |  |       this.indent = indent; | 
					
						
							|  |  |  |       this.prev = prev; | 
					
						
							|  |  |  |       this.line = line || {firstWord: "", indent: 0}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function pushContext(state, stream, type, indent) { | 
					
						
							|  |  |  |       indent = indent >= 0 ? indent : indentUnit; | 
					
						
							|  |  |  |       state.context = new Context(type, stream.indentation() + indent, state.context); | 
					
						
							|  |  |  |       return type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function popContext(state, currentIndent) { | 
					
						
							|  |  |  |       var contextIndent = state.context.indent - indentUnit; | 
					
						
							|  |  |  |       currentIndent = currentIndent || false; | 
					
						
							|  |  |  |       state.context = state.context.prev; | 
					
						
							|  |  |  |       if (currentIndent) state.context.indent = contextIndent; | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function pass(type, stream, state) { | 
					
						
							|  |  |  |       return states[state.context.type](type, stream, state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function popAndPass(type, stream, state, n) { | 
					
						
							|  |  |  |       for (var i = n || 1; i > 0; i--) | 
					
						
							|  |  |  |         state.context = state.context.prev; | 
					
						
							|  |  |  |       return pass(type, stream, state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Parser | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function wordIsTag(word) { | 
					
						
							|  |  |  |       return word.toLowerCase() in tagKeywords; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function wordIsProperty(word) { | 
					
						
							|  |  |  |       word = word.toLowerCase(); | 
					
						
							|  |  |  |       return word in propertyKeywords || word in fontProperties; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function wordIsBlock(word) { | 
					
						
							|  |  |  |       return word.toLowerCase() in blockKeywords; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function wordIsVendorPrefix(word) { | 
					
						
							|  |  |  |       return word.toLowerCase().match(vendorPrefixesRegexp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function wordAsValue(word) { | 
					
						
							|  |  |  |       var wordLC = word.toLowerCase(); | 
					
						
							|  |  |  |       var override = "variable-2"; | 
					
						
							|  |  |  |       if (wordIsTag(word)) override = "tag"; | 
					
						
							|  |  |  |       else if (wordIsBlock(word)) override = "block-keyword"; | 
					
						
							|  |  |  |       else if (wordIsProperty(word)) override = "property"; | 
					
						
							|  |  |  |       else if (wordLC in valueKeywords || wordLC in commonAtoms) override = "atom"; | 
					
						
							|  |  |  |       else if (wordLC == "return" || wordLC in colorKeywords) override = "keyword"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Font family
 | 
					
						
							|  |  |  |       else if (word.match(/^[A-Z]/)) override = "string"; | 
					
						
							|  |  |  |       return override; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function typeIsBlock(type, stream) { | 
					
						
							|  |  |  |       return ((endOfLine(stream) && (type == "{" || type == "]" || type == "hash" || type == "qualifier")) || type == "block-mixin"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function typeIsInterpolation(type, stream) { | 
					
						
							|  |  |  |       return type == "{" && stream.match(/^\s*\$?[\w-]+/i, false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function typeIsPseudo(type, stream) { | 
					
						
							|  |  |  |       return type == ":" && stream.match(/^[a-z-]+/, false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function startOfLine(stream) { | 
					
						
							|  |  |  |       return stream.sol() || stream.string.match(new RegExp("^\\s*" + escapeRegExp(stream.current()))); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function endOfLine(stream) { | 
					
						
							|  |  |  |       return stream.eol() || stream.match(/^\s*$/, false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function firstWordOfLine(line) { | 
					
						
							|  |  |  |       var re = /^\s*[-_]*[a-z0-9]+[\w-]*/i; | 
					
						
							|  |  |  |       var result = typeof line == "string" ? line.match(re) : line.string.match(re); | 
					
						
							|  |  |  |       return result ? result[0].replace(/^\s*/, "") : ""; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Block | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.block = function(type, stream, state) { | 
					
						
							|  |  |  |       if ((type == "comment" && startOfLine(stream)) || | 
					
						
							|  |  |  |           (type == "," && endOfLine(stream)) || | 
					
						
							|  |  |  |           type == "mixin") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (typeIsInterpolation(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "interpolation"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (endOfLine(stream) && type == "]") { | 
					
						
							|  |  |  |         if (!/^\s*(\.|#|:|\[|\*|&)/.test(stream.string) && !wordIsTag(firstWordOfLine(stream))) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (typeIsBlock(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "}" && endOfLine(stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "variable-name") { | 
					
						
							|  |  |  |         if (stream.string.match(/^\s?\$[\w-\.\[\]\'\"]+$/) || wordIsBlock(firstWordOfLine(stream))) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "variableName"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           return pushContext(state, stream, "variableName", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "=") { | 
					
						
							|  |  |  |         if (!endOfLine(stream) && !wordIsBlock(firstWordOfLine(stream))) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "*") { | 
					
						
							|  |  |  |         if (endOfLine(stream) || stream.match(/\s*(,|\.|#|\[|:|{)/,false)) { | 
					
						
							|  |  |  |           override = "tag"; | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (typeIsPseudo(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "pseudo"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (/@(font-face|media|supports|(-moz-)?document)/.test(type)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, endOfLine(stream) ? "block" : "atBlock"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (/@(-(moz|ms|o|webkit)-)?keyframes$/.test(type)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "keyframes"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (/@extends?/.test(type)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "extend", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type && type.charAt(0) == "@") { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Property Lookup
 | 
					
						
							|  |  |  |         if (stream.indentation() > 0 && wordIsProperty(stream.current().slice(1))) { | 
					
						
							|  |  |  |           override = "variable-2"; | 
					
						
							|  |  |  |           return "block"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (/(@import|@require|@charset)/.test(type)) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "reference" && endOfLine(stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "(") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "parens"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (type == "vendor-prefixes") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "vendorPrefixes"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         var word = stream.current(); | 
					
						
							|  |  |  |         override = wordAsValue(word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (override == "property") { | 
					
						
							|  |  |  |           if (startOfLine(stream)) { | 
					
						
							|  |  |  |             return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             override = "atom"; | 
					
						
							|  |  |  |             return "block"; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (override == "tag") { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // tag is a css value
 | 
					
						
							|  |  |  |           if (/embed|menu|pre|progress|sub|table/.test(word)) { | 
					
						
							|  |  |  |             if (wordIsProperty(firstWordOfLine(stream))) { | 
					
						
							|  |  |  |               override = "atom"; | 
					
						
							|  |  |  |               return "block"; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // tag is an attribute
 | 
					
						
							|  |  |  |           if (stream.string.match(new RegExp("\\[\\s*" + word + "|" + word +"\\s*\\]"))) { | 
					
						
							|  |  |  |             override = "atom"; | 
					
						
							|  |  |  |             return "block"; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // tag is a variable
 | 
					
						
							|  |  |  |           if (tagVariablesRegexp.test(word)) { | 
					
						
							|  |  |  |             if ((startOfLine(stream) && stream.string.match(/=/)) || | 
					
						
							|  |  |  |                 (!startOfLine(stream) && | 
					
						
							|  |  |  |                  !stream.string.match(/^(\s*\.|#|\&|\[|\/|>|\*)/) && | 
					
						
							|  |  |  |                  !wordIsTag(firstWordOfLine(stream)))) { | 
					
						
							|  |  |  |               override = "variable-2"; | 
					
						
							|  |  |  |               if (wordIsBlock(firstWordOfLine(stream)))  return "block"; | 
					
						
							|  |  |  |               return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (endOfLine(stream)) return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (override == "block-keyword") { | 
					
						
							|  |  |  |           override = "keyword"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // Postfix conditionals
 | 
					
						
							|  |  |  |           if (stream.current(/(if|unless)/) && !startOfLine(stream)) { | 
					
						
							|  |  |  |             return "block"; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (word == "return") return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Placeholder selector
 | 
					
						
							|  |  |  |         if (override == "variable-2" && stream.string.match(/^\s?\$[\w-\.\[\]\'\"]+$/)) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Parens | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.parens = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "(") return pushContext(state, stream, "parens"); | 
					
						
							|  |  |  |       if (type == ")") { | 
					
						
							|  |  |  |         if (state.context.prev.type == "parens") { | 
					
						
							|  |  |  |           return popContext(state); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ((stream.string.match(/^[a-z][\w-]*\(/i) && endOfLine(stream)) || | 
					
						
							|  |  |  |             wordIsBlock(firstWordOfLine(stream)) || | 
					
						
							|  |  |  |             /(\.|#|:|\[|\*|&|>|~|\+|\/)/.test(firstWordOfLine(stream)) || | 
					
						
							|  |  |  |             (!stream.string.match(/^-?[a-z][\w-\.\[\]\'\"]*\s*=/) && | 
					
						
							|  |  |  |              wordIsTag(firstWordOfLine(stream)))) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (stream.string.match(/^[\$-]?[a-z][\w-\.\[\]\'\"]*\s*=/) || | 
					
						
							|  |  |  |             stream.string.match(/^\s*(\(|\)|[0-9])/) || | 
					
						
							|  |  |  |             stream.string.match(/^\s+[a-z][\w-]*\(/i) || | 
					
						
							|  |  |  |             stream.string.match(/^\s+[\$-]?[a-z]/i)) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (endOfLine(stream)) return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         else return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type && type.charAt(0) == "@" && wordIsProperty(stream.current().slice(1))) { | 
					
						
							|  |  |  |         override = "variable-2"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         var word = stream.current(); | 
					
						
							|  |  |  |         override = wordAsValue(word); | 
					
						
							|  |  |  |         if (override == "tag" && tagVariablesRegexp.test(word)) { | 
					
						
							|  |  |  |           override = "variable-2"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (override == "property" || word == "to") override = "atom"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "variable-name") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "variableName"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (typeIsPseudo(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "pseudo"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Vendor prefixes | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.vendorPrefixes = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         override = "property"; | 
					
						
							|  |  |  |         return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return popContext(state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Pseudo | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.pseudo = function(type, stream, state) { | 
					
						
							|  |  |  |       if (!wordIsProperty(firstWordOfLine(stream.string))) { | 
					
						
							|  |  |  |         stream.match(/^[a-z-]+/); | 
					
						
							|  |  |  |         override = "variable-3"; | 
					
						
							|  |  |  |         if (endOfLine(stream)) return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         return popContext(state); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return popAndPass(type, stream, state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * atBlock | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.atBlock = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "(") return pushContext(state, stream, "atBlock_parens"); | 
					
						
							|  |  |  |       if (typeIsBlock(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (typeIsInterpolation(type, stream)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "interpolation"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         var word = stream.current().toLowerCase(); | 
					
						
							|  |  |  |         if (/^(only|not|and|or)$/.test(word)) | 
					
						
							|  |  |  |           override = "keyword"; | 
					
						
							|  |  |  |         else if (documentTypes.hasOwnProperty(word)) | 
					
						
							|  |  |  |           override = "tag"; | 
					
						
							|  |  |  |         else if (mediaTypes.hasOwnProperty(word)) | 
					
						
							|  |  |  |           override = "attribute"; | 
					
						
							|  |  |  |         else if (mediaFeatures.hasOwnProperty(word)) | 
					
						
							|  |  |  |           override = "property"; | 
					
						
							|  |  |  |         else if (nonStandardPropertyKeywords.hasOwnProperty(word)) | 
					
						
							|  |  |  |           override = "string-2"; | 
					
						
							|  |  |  |         else override = wordAsValue(stream.current()); | 
					
						
							|  |  |  |         if (override == "tag" && endOfLine(stream)) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "operator" && /^(not|and|or)$/.test(stream.current())) { | 
					
						
							|  |  |  |         override = "keyword"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     states.atBlock_parens = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "{" || type == "}") return state.context.type; | 
					
						
							|  |  |  |       if (type == ")") { | 
					
						
							|  |  |  |         if (endOfLine(stream)) return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         else return pushContext(state, stream, "atBlock"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         var word = stream.current().toLowerCase(); | 
					
						
							|  |  |  |         override = wordAsValue(word); | 
					
						
							|  |  |  |         if (/^(max|min)/.test(word)) override = "property"; | 
					
						
							|  |  |  |         if (override == "tag") { | 
					
						
							|  |  |  |           tagVariablesRegexp.test(word) ? override = "variable-2" : override = "atom"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return state.context.type; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return states.atBlock(type, stream, state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Keyframes | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.keyframes = function(type, stream, state) { | 
					
						
							|  |  |  |       if (stream.indentation() == "0" && ((type == "}" && startOfLine(stream)) || type == "]" || type == "hash" | 
					
						
							|  |  |  |                                           || type == "qualifier" || wordIsTag(stream.current()))) { | 
					
						
							|  |  |  |         return popAndPass(type, stream, state); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "{") return pushContext(state, stream, "keyframes"); | 
					
						
							|  |  |  |       if (type == "}") { | 
					
						
							|  |  |  |         if (startOfLine(stream)) return popContext(state, true); | 
					
						
							|  |  |  |         else return pushContext(state, stream, "keyframes"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "unit" && /^[0-9]+\%$/.test(stream.current())) { | 
					
						
							|  |  |  |         return pushContext(state, stream, "keyframes"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         override = wordAsValue(stream.current()); | 
					
						
							|  |  |  |         if (override == "block-keyword") { | 
					
						
							|  |  |  |           override = "keyword"; | 
					
						
							|  |  |  |           return pushContext(state, stream, "keyframes"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (/@(font-face|media|supports|(-moz-)?document)/.test(type)) { | 
					
						
							|  |  |  |         return pushContext(state, stream, endOfLine(stream) ? "block" : "atBlock"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "mixin") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Interpolation | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.interpolation = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "{") popContext(state) && pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       if (type == "}") { | 
					
						
							|  |  |  |         if (stream.string.match(/^\s*(\.|#|:|\[|\*|&|>|~|\+|\/)/i) || | 
					
						
							|  |  |  |             (stream.string.match(/^\s*[a-z]/i) && wordIsTag(firstWordOfLine(stream)))) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!stream.string.match(/^(\{|\s*\&)/) || | 
					
						
							|  |  |  |             stream.match(/\s*[\w-]/,false)) { | 
					
						
							|  |  |  |           return pushContext(state, stream, "block", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return pushContext(state, stream, "block"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "variable-name") { | 
					
						
							|  |  |  |         return pushContext(state, stream, "variableName", 0); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         override = wordAsValue(stream.current()); | 
					
						
							|  |  |  |         if (override == "tag") override = "atom"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return state.context.type; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Extend/s | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.extend = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "[" || type == "=") return "extend"; | 
					
						
							|  |  |  |       if (type == "]") return popContext(state); | 
					
						
							|  |  |  |       if (type == "word") { | 
					
						
							|  |  |  |         override = wordAsValue(stream.current()); | 
					
						
							|  |  |  |         return "extend"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return popContext(state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Variable name | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     states.variableName = function(type, stream, state) { | 
					
						
							|  |  |  |       if (type == "string" || type == "[" || type == "]" || stream.current().match(/^(\.|\$)/)) { | 
					
						
							|  |  |  |         if (stream.current().match(/^\.[\w-]+/i)) override = "variable-2"; | 
					
						
							|  |  |  |         return "variableName"; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return popAndPass(type, stream, state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       startState: function(base) { | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |           tokenize: null, | 
					
						
							|  |  |  |           state: "block", | 
					
						
							|  |  |  |           context: new Context("block", base || 0, null) | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       token: function(stream, state) { | 
					
						
							|  |  |  |         if (!state.tokenize && stream.eatSpace()) return null; | 
					
						
							|  |  |  |         style = (state.tokenize || tokenBase)(stream, state); | 
					
						
							|  |  |  |         if (style && typeof style == "object") { | 
					
						
							|  |  |  |           type = style[1]; | 
					
						
							|  |  |  |           style = style[0]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         override = style; | 
					
						
							|  |  |  |         state.state = states[state.state](type, stream, state); | 
					
						
							|  |  |  |         return override; | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       indent: function(state, textAfter, line) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         var cx = state.context, | 
					
						
							|  |  |  |             ch = textAfter && textAfter.charAt(0), | 
					
						
							|  |  |  |             indent = cx.indent, | 
					
						
							|  |  |  |             lineFirstWord = firstWordOfLine(textAfter), | 
					
						
							|  |  |  |             lineIndent = line.match(/^\s*/)[0].replace(/\t/g, indentUnitString).length, | 
					
						
							|  |  |  |             prevLineFirstWord = state.context.prev ? state.context.prev.line.firstWord : "", | 
					
						
							|  |  |  |             prevLineIndent = state.context.prev ? state.context.prev.line.indent : lineIndent; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cx.prev && | 
					
						
							|  |  |  |             (ch == "}" && (cx.type == "block" || cx.type == "atBlock" || cx.type == "keyframes") || | 
					
						
							|  |  |  |              ch == ")" && (cx.type == "parens" || cx.type == "atBlock_parens") || | 
					
						
							|  |  |  |              ch == "{" && (cx.type == "at"))) { | 
					
						
							|  |  |  |           indent = cx.indent - indentUnit; | 
					
						
							|  |  |  |         } else if (!(/(\})/.test(ch))) { | 
					
						
							|  |  |  |           if (/@|\$|\d/.test(ch) || | 
					
						
							|  |  |  |               /^\{/.test(textAfter) || | 
					
						
							|  |  |  | /^\s*\/(\/|\*)/.test(textAfter) || | 
					
						
							|  |  |  |               /^\s*\/\*/.test(prevLineFirstWord) || | 
					
						
							|  |  |  |               /^\s*[\w-\.\[\]\'\"]+\s*(\?|:|\+)?=/i.test(textAfter) || | 
					
						
							|  |  |  | /^(\+|-)?[a-z][\w-]*\(/i.test(textAfter) || | 
					
						
							|  |  |  | /^return/.test(textAfter) || | 
					
						
							|  |  |  |               wordIsBlock(lineFirstWord)) { | 
					
						
							|  |  |  |             indent = lineIndent; | 
					
						
							|  |  |  |           } else if (/(\.|#|:|\[|\*|&|>|~|\+|\/)/.test(ch) || wordIsTag(lineFirstWord)) { | 
					
						
							|  |  |  |             if (/\,\s*$/.test(prevLineFirstWord)) { | 
					
						
							|  |  |  |               indent = prevLineIndent; | 
					
						
							|  |  |  |             } else if (/^\s+/.test(line) && (/(\.|#|:|\[|\*|&|>|~|\+|\/)/.test(prevLineFirstWord) || wordIsTag(prevLineFirstWord))) { | 
					
						
							|  |  |  |               indent = lineIndent <= prevLineIndent ? prevLineIndent : prevLineIndent + indentUnit; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |               indent = lineIndent; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } else if (!/,\s*$/.test(line) && (wordIsVendorPrefix(lineFirstWord) || wordIsProperty(lineFirstWord))) { | 
					
						
							|  |  |  |             if (wordIsBlock(prevLineFirstWord)) { | 
					
						
							|  |  |  |               indent = lineIndent <= prevLineIndent ? prevLineIndent : prevLineIndent + indentUnit; | 
					
						
							|  |  |  |             } else if (/^\{/.test(prevLineFirstWord)) { | 
					
						
							|  |  |  |               indent = lineIndent <= prevLineIndent ? lineIndent : prevLineIndent + indentUnit; | 
					
						
							|  |  |  |             } else if (wordIsVendorPrefix(prevLineFirstWord) || wordIsProperty(prevLineFirstWord)) { | 
					
						
							|  |  |  |               indent = lineIndent >= prevLineIndent ? prevLineIndent : lineIndent; | 
					
						
							|  |  |  |             } else if (/^(\.|#|:|\[|\*|&|@|\+|\-|>|~|\/)/.test(prevLineFirstWord) || | 
					
						
							|  |  |  |                       /=\s*$/.test(prevLineFirstWord) || | 
					
						
							|  |  |  |                       wordIsTag(prevLineFirstWord) || | 
					
						
							|  |  |  |                       /^\$[\w-\.\[\]\'\"]/.test(prevLineFirstWord)) { | 
					
						
							|  |  |  |               indent = prevLineIndent + indentUnit; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |               indent = lineIndent; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return indent; | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       electricChars: "}", | 
					
						
							| 
									
										
										
										
											2021-04-06 22:16:34 +02:00
										 |  |  |       blockCommentStart: "/*", | 
					
						
							|  |  |  |       blockCommentEnd: "*/", | 
					
						
							|  |  |  |       blockCommentContinue: " * ", | 
					
						
							| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  |       lineComment: "//", | 
					
						
							|  |  |  |       fold: "indent" | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // developer.mozilla.org/en-US/docs/Web/HTML/Element
 | 
					
						
							|  |  |  |   var tagKeywords_ = ["a","abbr","address","area","article","aside","audio", "b", "base","bdi", "bdo","bgsound","blockquote","body","br","button","canvas","caption","cite", "code","col","colgroup","data","datalist","dd","del","details","dfn","div", "dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1", "h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe", "img","input","ins","kbd","keygen","label","legend","li","link","main","map", "mark","marquee","menu","menuitem","meta","meter","nav","nobr","noframes", "noscript","object","ol","optgroup","option","output","p","param","pre", "progress","q","rp","rt","ruby","s","samp","script","section","select", "small","source","span","strong","style","sub","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","track", "u","ul","var","video"]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // github.com/codemirror/CodeMirror/blob/master/mode/css/css.js
 | 
					
						
							| 
									
										
										
										
											2021-04-06 22:16:34 +02:00
										 |  |  |   // Note, "url-prefix" should precede "url" in order to match correctly in documentTypesRegexp
 | 
					
						
							|  |  |  |   var documentTypes_ = ["domain", "regexp", "url-prefix", "url"]; | 
					
						
							| 
									
										
										
										
											2018-01-21 10:33:32 -05:00
										 |  |  |   var mediaTypes_ = ["all","aural","braille","handheld","print","projection","screen","tty","tv","embossed"]; | 
					
						
							|  |  |  |   var mediaFeatures_ = ["width","min-width","max-width","height","min-height","max-height","device-width","min-device-width","max-device-width","device-height","min-device-height","max-device-height","aspect-ratio","min-aspect-ratio","max-aspect-ratio","device-aspect-ratio","min-device-aspect-ratio","max-device-aspect-ratio","color","min-color","max-color","color-index","min-color-index","max-color-index","monochrome","min-monochrome","max-monochrome","resolution","min-resolution","max-resolution","scan","grid"]; | 
					
						
							|  |  |  |   var propertyKeywords_ = ["align-content","align-items","align-self","alignment-adjust","alignment-baseline","anchor-point","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","appearance","azimuth","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","baseline-shift","binding","bleed","bookmark-label","bookmark-level","bookmark-state","bookmark-target","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","color","color-profile","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","crop","cue","cue-after","cue-before","cursor","direction","display","dominant-baseline","drop-initial-after-adjust","drop-initial-after-align","drop-initial-before-adjust","drop-initial-before-align","drop-initial-size","drop-initial-value","elevation","empty-cells","fit","fit-position","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","float-offset","flow-from","flow-into","font","font-feature-settings","font-family","font-kerning","font-language-override","font-size","font-size-adjust","font-stretch","font-style","font-synthesis","font-variant","font-variant-alternates","font-variant-caps","font-variant-east-asian","font-variant-ligatures","font-variant-numeric","font-variant-position","font-weight","grid","grid-area","grid-auto-columns","grid-auto-flow","grid-auto-position","grid-auto-rows","grid-column","grid-column-end","grid-column-start","grid-row","grid-row-end","grid-row-start","grid-template","grid-template-areas","grid-template-columns","grid-template-rows","hanging-punctuation","height","hyphens","icon","image-orientation","image-rendering","image-resolution","inline-box-align","justify-content","left","letter-spacing","line-break","line-height","line-stacking","line-stacking-ruby","line-stacking-shift","line-stacking-strategy","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marker-offset","marks","marquee-direction","marquee-loop","marquee-play-count","marquee-speed","marquee-style","max-height","max-width","min-height","min-width","move-to","nav-down","nav-index","nav-left","nav-right","nav-up","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-style","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page","page-break-after","page-break-before","page-break-inside","page-policy","pause","pause-after","pause-before","perspective","perspective-origin","pitch","pitch-range","play-during","position","presentation-level","punctuation-trim","quotes","region-break-after","region-break-before","region-break-inside","region-fragment","rendering-intent","resize","rest","rest-after","rest-before","richness","right","rotation","rotation-point","ruby-align","ruby-overhang","ruby-position","ruby-span","shape-image-th | 
					
						
							|  |  |  |   var nonStandardPropertyKeywords_ = ["scrollbar-arrow-color","scrollbar-base-color","scrollbar-dark-shadow-color","scrollbar-face-color","scrollbar-highlight-color","scrollbar-shadow-color","scrollbar-3d-light-color","scrollbar-track-color","shape-inside","searchfield-cancel-button","searchfield-decoration","searchfield-results-button","searchfield-results-decoration","zoom"]; | 
					
						
							|  |  |  |   var fontProperties_ = ["font-family","src","unicode-range","font-variant","font-feature-settings","font-stretch","font-weight","font-style"]; | 
					
						
							|  |  |  |   var colorKeywords_ = ["aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","grey","green","greenyellow","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen"]; | 
					
						
							|  |  |  |   var valueKeywords_ = ["above","absolute","activeborder","additive","activecaption","afar","after-white-space","ahead","alias","all","all-scroll","alphabetic","alternate","always","amharic","amharic-abegede","antialiased","appworkspace","arabic-indic","armenian","asterisks","attr","auto","avoid","avoid-column","avoid-page","avoid-region","background","backwards","baseline","below","bidi-override","binary","bengali","blink","block","block-axis","bold","bolder","border","border-box","both","bottom","break","break-all","break-word","bullets","button","button-bevel","buttonface","buttonhighlight","buttonshadow","buttontext","calc","cambodian","capitalize","caps-lock-indicator","caption","captiontext","caret","cell","center","checkbox","circle","cjk-decimal","cjk-earthly-branch","cjk-heavenly-stem","cjk-ideographic","clear","clip","close-quote","col-resize","collapse","column","compact","condensed","contain","content","contents","content-box","context-menu","continuous","copy","counter","counters","cover","crop","cross","crosshair","currentcolor","cursive","cyclic","dashed","decimal","decimal-leading-zero","default","default-button","destination-atop","destination-in","destination-out","destination-over","devanagari","disc","discard","disclosure-closed","disclosure-open","document","dot-dash","dot-dot-dash","dotted","double","down","e-resize","ease","ease-in","ease-in-out","ease-out","element","ellipse","ellipsis","embed","end","ethiopic","ethiopic-abegede","ethiopic-abegede-am-et","ethiopic-abegede-gez","ethiopic-abegede-ti-er","ethiopic-abegede-ti-et","ethiopic-halehame-aa-er","ethiopic-halehame-aa-et","ethiopic-halehame-am-et","ethiopic-halehame-gez","ethiopic-halehame-om-et","ethiopic-halehame-sid-et","ethiopic-halehame-so-et","ethiopic-halehame-ti-er","ethiopic-halehame-ti-et","ethiopic-halehame-tig","ethiopic-numeric","ew-resize","expanded","extends","extra-condensed","extra-expanded","fantasy","fast","fill","fixed","flat","flex","footnotes","forwards","from","geometricPrecision","georgian","graytext","groove","gujarati","gurmukhi","hand","hangul","hangul-consonant","hebrew","help","hidden","hide","higher","highlight","highlighttext","hiragana","hiragana-iroha","horizontal","hsl","hsla","icon","ignore","inactiveborder","inactivecaption","inactivecaptiontext","infinite","infobackground","infotext","inherit","initial","inline","inline-axis","inline-block","inline-flex","inline-table","inset","inside","intrinsic","invert","italic","japanese-formal","japanese-informal","justify","kannada","katakana","katakana-iroha","keep-all","khmer","korean-hangul-formal","korean-hanja-formal","korean-hanja-informal","landscape","lao","large","larger","left","level","lighter","line-through","linear","linear-gradient","lines","list-item","listbox","listitem","local","logical","loud","lower","lower-alpha","lower-armenian","lower-greek","lower-hexadecimal","lower-latin","lower-norwegian","lower-roman","lowercase","ltr","malayalam","match","matrix","matrix3d","media-controls-background","media-current-time-display","media-fullscreen-button","media-mute-button","media-play-button","media-return-to-realtime-button","media-rewind-button","media-seek-back-button","media-seek-forward-button","media-slider","media-sliderthumb","media-time-remaining-display","media-volume-slider","media-volume-slider-container","media-volume-sliderthumb","medium","menu","menulist","menulist-button","menulist-text","menulist-textfield","menutext","message-box","middle","min-intrinsic","mix","mongolian","monospace","move","multiple","myanmar","n-resize","narrower","ne-resize","nesw-resize","no-close-quote","no-drop","no-open-quote","no-repeat","none","normal","not-allowed","nowrap","ns-resize","numbers","numeric","nw-resize","nwse-resize","oblique","octal","open-quote","optimizeLegibility","optimizeSpeed","oriya","oromo","outset","outside","outside-shape","overlay","overline","padding","padding-box","painted","page","paused","persian","perspective","plus-darker","plus-lighter","pointer","polygon","portrait","pre","pre-line","pre-wrap","preserve-3d","progress","push- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var wordOperatorKeywords_ = ["in","and","or","not","is not","is a","is","isnt","defined","if unless"], | 
					
						
							|  |  |  |       blockKeywords_ = ["for","if","else","unless", "from", "to"], | 
					
						
							|  |  |  |       commonAtoms_ = ["null","true","false","href","title","type","not-allowed","readonly","disabled"], | 
					
						
							|  |  |  |       commonDef_ = ["@font-face", "@keyframes", "@media", "@viewport", "@page", "@host", "@supports", "@block", "@css"]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var hintWords = tagKeywords_.concat(documentTypes_,mediaTypes_,mediaFeatures_, | 
					
						
							|  |  |  |                                       propertyKeywords_,nonStandardPropertyKeywords_, | 
					
						
							|  |  |  |                                       colorKeywords_,valueKeywords_,fontProperties_, | 
					
						
							|  |  |  |                                       wordOperatorKeywords_,blockKeywords_, | 
					
						
							|  |  |  |                                       commonAtoms_,commonDef_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   function wordRegexp(words) { | 
					
						
							|  |  |  |     words = words.sort(function(a,b){return b > a;}); | 
					
						
							|  |  |  |     return new RegExp("^((" + words.join(")|(") + "))\\b"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   function keySet(array) { | 
					
						
							|  |  |  |     var keys = {}; | 
					
						
							|  |  |  |     for (var i = 0; i < array.length; ++i) keys[array[i]] = true; | 
					
						
							|  |  |  |     return keys; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   function escapeRegExp(text) { | 
					
						
							|  |  |  |     return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CodeMirror.registerHelper("hintWords", "stylus", hintWords); | 
					
						
							|  |  |  |   CodeMirror.defineMIME("text/x-styl", "stylus"); | 
					
						
							|  |  |  | }); |