// CodeMirror, copyright (c) by Marijn Haverbeke and others // Distributed under an MIT license: http://codemirror.net/LICENSE // Swift mode created by Michael Kaminsky https://github.com/mkaminsky11 (function(mod) { if (typeof exports == "object" && typeof module == "object") mod(require("../../lib/codemirror")) else if (typeof define == "function" && define.amd) define(["../../lib/codemirror"], mod) else mod(CodeMirror) })(function(CodeMirror) { "use strict" function trim(str) { return /^\s*(.*?)\s*$/.exec(str)[1] } var separators = [" ","\\\+","\\\-","\\\(","\\\)","\\\*","/",":","\\\?","\\\<","\\\>"," ","\\\."] var tokens = new RegExp(separators.join("|"),"g") function getWord(string, pos) { var index = -1, count = 1 var words = string.split(tokens) for (var i = 0; i < words.length; i++) { for(var j = 1; j <= words[i].length; j++) { if (count==pos) index = i count++ } count++ } var ret = ["", ""] if (pos == 0) { ret[1] = words[0] ret[0] = null } else { ret[1] = words[index] ret[0] = words[index-1] } return ret } CodeMirror.defineMode("swift", function() { var keywords=["var","let","class","deinit","enum","extension","func","import","init","let","protocol","static","struct","subscript","typealias","var","as","dynamicType","is","new","super","self","Self","Type","__COLUMN__","__FILE__","__FUNCTION__","__LINE__","break","case","continue","default","do","else","fallthrough","if","in","for","return","switch","where","while","associativity","didSet","get","infix","inout","left","mutating","none","nonmutating","operator","override","postfix","precedence","prefix","right","set","unowned","unowned(safe)","unowned(unsafe)","weak","willSet"] var commonConstants=["Infinity","NaN","undefined","null","true","false","on","off","yes","no","nil","null","this","super"] var types=["String","bool","int","string","double","Double","Int","Float","float","public","private","extension"] var numbers=["0","1","2","3","4","5","6","7","8","9"] var operators=["+","-","/","*","%","=","|","&","<",">"] var punc=[";",",",".","(",")","{","}","[","]"] var delimiters=/^(?:[()\[\]{},:`=;]|\.\.?\.?)/ var identifiers=/^[_A-Za-z$][_A-Za-z$0-9]*/ var properties=/^(@|this\.)[_A-Za-z$][_A-Za-z$0-9]*/ var regexPrefixes=/^(\/{3}|\/)/ return { startState: function() { return { prev: false, string: false, escape: false, inner: false, comment: false, num_left: 0, num_right: 0, doubleString: false, singleString: false } }, token: function(stream, state) { if (stream.eatSpace()) return null var ch = stream.next() if (state.string) { if (state.escape) { state.escape = false return "string" } else { if ((ch == "\"" && (state.doubleString && !state.singleString) || (ch == "'" && (!state.doubleString && state.singleString))) && !state.escape) { state.string = false state.doubleString = false state.singleString = false return "string" } else if (ch == "\\" && stream.peek() == "(") { state.inner = true state.string = false return "keyword" } else if (ch == "\\" && stream.peek() != "(") { state.escape = true state.string = true return "string" } else { return "string" } } } else if (state.comment) { if (ch == "*" && stream.peek() == "/") { state.prev = "*" return "comment" } else if (ch == "/" && state.prev == "*") { state.prev = false state.comment = false return "comment" } return "comment" } else { if (ch == "/") { if (stream.peek() == "/") { stream.skipToEnd() return "comment" } if (stream.peek() == "*") { state.comment = true return "comment" } } if (ch == "(" && state.inner) { state.num_left++ return null } if (ch == ")" && state.inner) { state.num_right++ if (state.num_left == state.num_right) { state.inner=false state.string=true } return null } var ret = getWord(stream.string, stream.pos) var the_word = ret[1] var prev_word = ret[0] if (operators.indexOf(ch + "") > -1) return "operator" if (punc.indexOf(ch) > -1) return "punctuation" if (typeof the_word != "undefined") { the_word = trim(the_word) if (typeof prev_word != "undefined") prev_word = trim(prev_word) if (the_word.charAt(0) == "#") return null if (types.indexOf(the_word) > -1) return "def" if (commonConstants.indexOf(the_word) > -1) return "atom" if (numbers.indexOf(the_word) > -1) return "number" if ((numbers.indexOf(the_word.charAt(0) + "") > -1 || operators.indexOf(the_word.charAt(0) + "") > -1) && numbers.indexOf(ch) > -1) { return "number" } if (keywords.indexOf(the_word) > -1 || keywords.indexOf(the_word.split(tokens)[0]) > -1) return "keyword" if (keywords.indexOf(prev_word) > -1) return "def" } if (ch == '"' && !state.doubleString) { state.string = true state.doubleString = true return "string" } if (ch == "'" && !state.singleString) { state.string = true state.singleString = true return "string" } if (ch == "(" && state.inner) state.num_left++ if (ch == ")" && state.inner) { state.num_right++ if (state.num_left == state.num_right) { state.inner = false state.string = true } return null } if (stream.match(/^-?[0-9\.]/, false)) { if (stream.match(/^-?\d*\.\d+(e[\+\-]?\d+)?/i) || stream.match(/^-?\d+\.\d*/) || stream.match(/^-?\.\d+/)) { if (stream.peek() == ".") stream.backUp(1) return "number" } if (stream.match(/^-?0x[0-9a-f]+/i) || stream.match(/^-?[1-9]\d*(e[\+\-]?\d+)?/) || stream.match(/^-?0(?![\dx])/i)) return "number" } if (stream.match(regexPrefixes)) { if (stream.current()!="/" || stream.match(/^.*\//,false)) return "string" else stream.backUp(1) } if (stream.match(delimiters)) return "punctuation" if (stream.match(identifiers)) return "variable" if (stream.match(properties)) return "property" return "variable" } } } }) CodeMirror.defineMIME("text/x-swift","swift") })