const CodeMirror = require('codemirror') CodeMirror.defineMode('kotlin', function(config, parserConfig) { function words(str) { var obj = {}, words = str.split(' ') for (var i = 0; i < words.length; ++i) obj[words[i]] = true return obj } var multiLineStrings = parserConfig.multiLineStrings var keywords = words( 'package continue return object while break class data trait throw super ' + 'when type this else This try val var fun for is in if do as true false null get set ' + 'import where by get set abstract enum open annotation override private public internal ' + 'protected catch out vararg inline finally final ref const' ) var blockKeywords = words('catch class do else finally for if where try while enum') var atoms = words('null true false this') var builtins = words( 'Int Double Float Long Short Byte IntArray ShortArray ByteArray String Boolean List Set ' + 'Map MutableList MutableSet print println shl shr ushr and or xor inv ' ) var curPunc function tokenBase(stream, state) { var ch = stream.next() if (ch == '"' || ch == "'") { return startString(ch, stream, state) } // Wildcard import w/o trailing semicolon (import smth.*) if (ch == '.' && stream.eat('*')) { return 'word' } if (/[\[\]{}\(\),;\:\.]/.test(ch)) { curPunc = ch return null } if (/\d/.test(ch)) { if (stream.eat(/eE/)) { stream.eat(/\+\-/) stream.eatWhile(/\d/) } return 'number' } if (ch == '/') { if (stream.eat('*')) { state.tokenize.push(tokenComment) return tokenComment(stream, state) } if (stream.eat('/')) { stream.skipToEnd() return 'comment' } if (expectExpression(state.lastToken)) { return startString(ch, stream, state) } } // Commented if (ch == '-' && stream.eat('>')) { curPunc = '->' return null } if (/[\-+*&%=<>!?|\/~]/.test(ch)) { stream.eatWhile(/[\-+*&%=<>|~]/) return 'operator' } stream.eatWhile(/[\w\$_]/) var cur = stream.current() if (atoms.propertyIsEnumerable(cur)) { return 'atom' } if (keywords.propertyIsEnumerable(cur)) { if (blockKeywords.propertyIsEnumerable(cur)) curPunc = 'newstatement' return 'keyword' } if (builtins.propertyIsEnumerable(cur)) { return 'builtin' } return 'word' } tokenBase.isBase = true function startString(quote, stream, state) { var tripleQuoted = false if (quote != '/' && stream.eat(quote)) { if (stream.eat(quote)) tripleQuoted = true else return 'string' } function t(stream, state) { var escaped = false, next, end = !tripleQuoted while ((next = stream.next()) != null) { if (next == quote && !escaped) { if (!tripleQuoted) { break } if (stream.match(quote + quote)) { end = true break } } if (quote == '"' && next == '$' && !escaped && stream.eat('{')) { state.tokenize.push(tokenBaseUntilBrace()) return 'string' } if (next == '$' && !escaped && !stream.eat(' ')) { state.tokenize.push(tokenBaseUntilSpace()) return 'string' } escaped = !escaped && next == '\\' } if (multiLineStrings) state.tokenize.push(t) if (end) state.tokenize.pop() return 'string' } state.tokenize.push(t) return t(stream, state) } function tokenBaseUntilBrace() { var depth = 1 function t(stream, state) { if (stream.peek() == '}') { depth-- if (depth == 0) { state.tokenize.pop() return state.tokenize[state.tokenize.length - 1](stream, state) } } else if (stream.peek() == '{') { depth++ } return tokenBase(stream, state) } t.isBase = true return t } function tokenBaseUntilSpace() { function t(stream, state) { if (stream.eat(/[\w]/)) { var isWord = stream.eatWhile(/[\w]/) if (isWord) { state.tokenize.pop() return 'word' } } state.tokenize.pop() return 'string' } t.isBase = true return t } function tokenComment(stream, state) { var maybeEnd = false, ch while ((ch = stream.next())) { if (ch == '/' && maybeEnd) { state.tokenize.pop() break } maybeEnd = ch == '*' } return 'comment' } function expectExpression(last) { return ( !last || last == 'operator' || last == '->' || /[\.\[\{\(,;:]/.test(last) || last == 'newstatement' || last == 'keyword' || last == 'proplabel' ) } function Context(indented, column, type, align, prev) { this.indented = indented this.column = column this.type = type this.align = align this.prev = prev } function pushContext(state, col, type) { return (state.context = new Context(state.indented, col, type, null, state.context)) } function popContext(state) { var t = state.context.type if (t == ')' || t == ']' || t == '}') state.indented = state.context.indented return (state.context = state.context.prev) } // Interface return { startState: function(basecolumn) { return { tokenize: [tokenBase], context: new Context((basecolumn || 0) - config.indentUnit, 0, 'top', false), indented: 0, startOfLine: true, lastToken: null } }, token: function(stream, state) { var ctx = state.context if (stream.sol()) { if (ctx.align == null) ctx.align = false state.indented = stream.indentation() state.startOfLine = true // Automatic semicolon insertion if (ctx.type == 'statement' && !expectExpression(state.lastToken)) { popContext(state) ctx = state.context } } if (stream.eatSpace()) return null curPunc = null var style = state.tokenize[state.tokenize.length - 1](stream, state) if (style == 'comment') return style if (ctx.align == null) ctx.align = true if ((curPunc == ';' || curPunc == ':') && ctx.type == 'statement') popContext(state) else if (curPunc == '->' && ctx.type == 'statement' && ctx.prev.type == '}') { // Handle indentation for {x -> \n ... } popContext(state) state.context.align = false } else if (curPunc == '{') pushContext(state, stream.column(), '}') else if (curPunc == '[') pushContext(state, stream.column(), ']') else if (curPunc == '(') pushContext(state, stream.column(), ')') else if (curPunc == '}') { while (ctx.type == 'statement') ctx = popContext(state) if (ctx.type == '}') ctx = popContext(state) while (ctx.type == 'statement') ctx = popContext(state) } else if (curPunc == ctx.type) popContext(state) else if ( ctx.type == '}' || ctx.type == 'top' || (ctx.type == 'statement' && curPunc == 'newstatement') ) pushContext(state, stream.column(), 'statement') state.startOfLine = false state.lastToken = curPunc || style return style }, indent: function(state, textAfter) { if (!state.tokenize[state.tokenize.length - 1].isBase) return 0 var firstChar = textAfter && textAfter.charAt(0), ctx = state.context if (ctx.type == 'statement' && !expectExpression(state.lastToken)) ctx = ctx.prev var closing = firstChar == ctx.type if (ctx.type == 'statement') { return ctx.indented + (firstChar == '{' ? 0 : config.indentUnit) } else if (ctx.align) return ctx.column + (closing ? 0 : 1) else return ctx.indented + (closing ? 0 : config.indentUnit) }, closeBrackets: { triples: '\'"' }, electricChars: '{}' } }) CodeMirror.defineMIME('text/x-kotlin', 'kotlin')