Added coding style modes to SimpleMDE (#3286)
* Added coding style modes to SimpleMDE * Moved the CodeMirror addon from simplemd to codemirror directorytokarchuk/v1.17
parent
cf85e9eb7b
commit
69dae1ec1c
@ -0,0 +1,8 @@ |
||||
*.txt text |
||||
*.js text |
||||
*.html text |
||||
*.md text |
||||
*.json text |
||||
*.yml text |
||||
*.css text |
||||
*.svg text |
@ -0,0 +1,8 @@ |
||||
/node_modules |
||||
/npm-debug.log |
||||
/test*.html |
||||
.tern-* |
||||
*~ |
||||
*.swp |
||||
.idea |
||||
*.iml |
@ -0,0 +1,10 @@ |
||||
/node_modules |
||||
/demo |
||||
/doc |
||||
/test |
||||
/test*.html |
||||
/index.html |
||||
/mode/*/*test.js |
||||
/mode/*/*.html |
||||
/mode/index.html |
||||
.* |
@ -0,0 +1,4 @@ |
||||
language: node_js |
||||
node_js: |
||||
- stable |
||||
sudo: false |
@ -0,0 +1,64 @@ |
||||
// 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"), "cjs"); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror"], function(CM) { mod(CM, "amd"); }); |
||||
else // Plain browser env
|
||||
mod(CodeMirror, "plain"); |
||||
})(function(CodeMirror, env) { |
||||
if (!CodeMirror.modeURL) CodeMirror.modeURL = "../mode/%N/%N.js"; |
||||
|
||||
var loading = {}; |
||||
function splitCallback(cont, n) { |
||||
var countDown = n; |
||||
return function() { if (--countDown == 0) cont(); }; |
||||
} |
||||
function ensureDeps(mode, cont) { |
||||
var deps = CodeMirror.modes[mode].dependencies; |
||||
if (!deps) return cont(); |
||||
var missing = []; |
||||
for (var i = 0; i < deps.length; ++i) { |
||||
if (!CodeMirror.modes.hasOwnProperty(deps[i])) |
||||
missing.push(deps[i]); |
||||
} |
||||
if (!missing.length) return cont(); |
||||
var split = splitCallback(cont, missing.length); |
||||
for (var i = 0; i < missing.length; ++i) |
||||
CodeMirror.requireMode(missing[i], split); |
||||
} |
||||
|
||||
CodeMirror.requireMode = function(mode, cont) { |
||||
if (typeof mode != "string") mode = mode.name; |
||||
if (CodeMirror.modes.hasOwnProperty(mode)) return ensureDeps(mode, cont); |
||||
if (loading.hasOwnProperty(mode)) return loading[mode].push(cont); |
||||
|
||||
var file = CodeMirror.modeURL.replace(/%N/g, mode); |
||||
if (env == "plain") { |
||||
var script = document.createElement("script"); |
||||
script.src = file; |
||||
var others = document.getElementsByTagName("script")[0]; |
||||
var list = loading[mode] = [cont]; |
||||
CodeMirror.on(script, "load", function() { |
||||
ensureDeps(mode, function() { |
||||
for (var i = 0; i < list.length; ++i) list[i](); |
||||
}); |
||||
}); |
||||
others.parentNode.insertBefore(script, others); |
||||
} else if (env == "cjs") { |
||||
require(file); |
||||
cont(); |
||||
} else if (env == "amd") { |
||||
requirejs([file], cont); |
||||
} |
||||
}; |
||||
|
||||
CodeMirror.autoLoadMode = function(instance, mode) { |
||||
if (!CodeMirror.modes.hasOwnProperty(mode)) |
||||
CodeMirror.requireMode(mode, function() { |
||||
instance.setOption("mode", instance.getOption("mode")); |
||||
}); |
||||
}; |
||||
}); |
@ -0,0 +1,123 @@ |
||||
// 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.multiplexingMode = function(outer /*, others */) { |
||||
// Others should be {open, close, mode [, delimStyle] [, innerStyle]} objects
|
||||
var others = Array.prototype.slice.call(arguments, 1); |
||||
|
||||
function indexOf(string, pattern, from, returnEnd) { |
||||
if (typeof pattern == "string") { |
||||
var found = string.indexOf(pattern, from); |
||||
return returnEnd && found > -1 ? found + pattern.length : found; |
||||
} |
||||
var m = pattern.exec(from ? string.slice(from) : string); |
||||
return m ? m.index + from + (returnEnd ? m[0].length : 0) : -1; |
||||
} |
||||
|
||||
return { |
||||
startState: function() { |
||||
return { |
||||
outer: CodeMirror.startState(outer), |
||||
innerActive: null, |
||||
inner: null |
||||
}; |
||||
}, |
||||
|
||||
copyState: function(state) { |
||||
return { |
||||
outer: CodeMirror.copyState(outer, state.outer), |
||||
innerActive: state.innerActive, |
||||
inner: state.innerActive && CodeMirror.copyState(state.innerActive.mode, state.inner) |
||||
}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (!state.innerActive) { |
||||
var cutOff = Infinity, oldContent = stream.string; |
||||
for (var i = 0; i < others.length; ++i) { |
||||
var other = others[i]; |
||||
var found = indexOf(oldContent, other.open, stream.pos); |
||||
if (found == stream.pos) { |
||||
if (!other.parseDelimiters) stream.match(other.open); |
||||
state.innerActive = other; |
||||
state.inner = CodeMirror.startState(other.mode, outer.indent ? outer.indent(state.outer, "") : 0); |
||||
return other.delimStyle && (other.delimStyle + " " + other.delimStyle + "-open"); |
||||
} else if (found != -1 && found < cutOff) { |
||||
cutOff = found; |
||||
} |
||||
} |
||||
if (cutOff != Infinity) stream.string = oldContent.slice(0, cutOff); |
||||
var outerToken = outer.token(stream, state.outer); |
||||
if (cutOff != Infinity) stream.string = oldContent; |
||||
return outerToken; |
||||
} else { |
||||
var curInner = state.innerActive, oldContent = stream.string; |
||||
if (!curInner.close && stream.sol()) { |
||||
state.innerActive = state.inner = null; |
||||
return this.token(stream, state); |
||||
} |
||||
var found = curInner.close ? indexOf(oldContent, curInner.close, stream.pos, curInner.parseDelimiters) : -1; |
||||
if (found == stream.pos && !curInner.parseDelimiters) { |
||||
stream.match(curInner.close); |
||||
state.innerActive = state.inner = null; |
||||
return curInner.delimStyle && (curInner.delimStyle + " " + curInner.delimStyle + "-close"); |
||||
} |
||||
if (found > -1) stream.string = oldContent.slice(0, found); |
||||
var innerToken = curInner.mode.token(stream, state.inner); |
||||
if (found > -1) stream.string = oldContent; |
||||
|
||||
if (found == stream.pos && curInner.parseDelimiters) |
||||
state.innerActive = state.inner = null; |
||||
|
||||
if (curInner.innerStyle) { |
||||
if (innerToken) innerToken = innerToken + " " + curInner.innerStyle; |
||||
else innerToken = curInner.innerStyle; |
||||
} |
||||
|
||||
return innerToken; |
||||
} |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
var mode = state.innerActive ? state.innerActive.mode : outer; |
||||
if (!mode.indent) return CodeMirror.Pass; |
||||
return mode.indent(state.innerActive ? state.inner : state.outer, textAfter); |
||||
}, |
||||
|
||||
blankLine: function(state) { |
||||
var mode = state.innerActive ? state.innerActive.mode : outer; |
||||
if (mode.blankLine) { |
||||
mode.blankLine(state.innerActive ? state.inner : state.outer); |
||||
} |
||||
if (!state.innerActive) { |
||||
for (var i = 0; i < others.length; ++i) { |
||||
var other = others[i]; |
||||
if (other.open === "\n") { |
||||
state.innerActive = other; |
||||
state.inner = CodeMirror.startState(other.mode, mode.indent ? mode.indent(state.outer, "") : 0); |
||||
} |
||||
} |
||||
} else if (state.innerActive.close === "\n") { |
||||
state.innerActive = state.inner = null; |
||||
} |
||||
}, |
||||
|
||||
electricChars: outer.electricChars, |
||||
|
||||
innerMode: function(state) { |
||||
return state.inner ? {state: state.inner, mode: state.innerActive.mode} : {state: state.outer, mode: outer}; |
||||
} |
||||
}; |
||||
}; |
||||
|
||||
}); |
@ -0,0 +1,33 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
CodeMirror.defineMode("markdown_with_stex", function(){ |
||||
var inner = CodeMirror.getMode({}, "stex"); |
||||
var outer = CodeMirror.getMode({}, "markdown"); |
||||
|
||||
var innerOptions = { |
||||
open: '$', |
||||
close: '$', |
||||
mode: inner, |
||||
delimStyle: 'delim', |
||||
innerStyle: 'inner' |
||||
}; |
||||
|
||||
return CodeMirror.multiplexingMode(outer, innerOptions); |
||||
}); |
||||
|
||||
var mode = CodeMirror.getMode({}, "markdown_with_stex"); |
||||
|
||||
function MT(name) { |
||||
test.mode( |
||||
name, |
||||
mode, |
||||
Array.prototype.slice.call(arguments, 1), |
||||
'multiplexing'); |
||||
} |
||||
|
||||
MT( |
||||
"stexInsideMarkdown", |
||||
"[strong **Equation:**] [delim&delim-open $][inner&tag \\pi][delim&delim-close $]"); |
||||
})(); |
@ -0,0 +1,85 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
// Utility function that allows modes to be combined. The mode given
|
||||
// as the base argument takes care of most of the normal mode
|
||||
// functionality, but a second (typically simple) mode is used, which
|
||||
// can override the style of text. Both modes get to parse all of the
|
||||
// text, but when both assign a non-null style to a piece of code, the
|
||||
// overlay wins, unless the combine argument was true and not overridden,
|
||||
// or state.overlay.combineTokens was true, in which case the styles are
|
||||
// combined.
|
||||
|
||||
(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.overlayMode = function(base, overlay, combine) { |
||||
return { |
||||
startState: function() { |
||||
return { |
||||
base: CodeMirror.startState(base), |
||||
overlay: CodeMirror.startState(overlay), |
||||
basePos: 0, baseCur: null, |
||||
overlayPos: 0, overlayCur: null, |
||||
streamSeen: null |
||||
}; |
||||
}, |
||||
copyState: function(state) { |
||||
return { |
||||
base: CodeMirror.copyState(base, state.base), |
||||
overlay: CodeMirror.copyState(overlay, state.overlay), |
||||
basePos: state.basePos, baseCur: null, |
||||
overlayPos: state.overlayPos, overlayCur: null |
||||
}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (stream != state.streamSeen || |
||||
Math.min(state.basePos, state.overlayPos) < stream.start) { |
||||
state.streamSeen = stream; |
||||
state.basePos = state.overlayPos = stream.start; |
||||
} |
||||
|
||||
if (stream.start == state.basePos) { |
||||
state.baseCur = base.token(stream, state.base); |
||||
state.basePos = stream.pos; |
||||
} |
||||
if (stream.start == state.overlayPos) { |
||||
stream.pos = stream.start; |
||||
state.overlayCur = overlay.token(stream, state.overlay); |
||||
state.overlayPos = stream.pos; |
||||
} |
||||
stream.pos = Math.min(state.basePos, state.overlayPos); |
||||
|
||||
// state.overlay.combineTokens always takes precedence over combine,
|
||||
// unless set to null
|
||||
if (state.overlayCur == null) return state.baseCur; |
||||
else if (state.baseCur != null && |
||||
state.overlay.combineTokens || |
||||
combine && state.overlay.combineTokens == null) |
||||
return state.baseCur + " " + state.overlayCur; |
||||
else return state.overlayCur; |
||||
}, |
||||
|
||||
indent: base.indent && function(state, textAfter) { |
||||
return base.indent(state.base, textAfter); |
||||
}, |
||||
electricChars: base.electricChars, |
||||
|
||||
innerMode: function(state) { return {state: state.base, mode: base}; }, |
||||
|
||||
blankLine: function(state) { |
||||
if (base.blankLine) base.blankLine(state.base); |
||||
if (overlay.blankLine) overlay.blankLine(state.overlay); |
||||
} |
||||
}; |
||||
}; |
||||
|
||||
}); |
@ -0,0 +1,213 @@ |
||||
// 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.defineSimpleMode = function(name, states) { |
||||
CodeMirror.defineMode(name, function(config) { |
||||
return CodeMirror.simpleMode(config, states); |
||||
}); |
||||
}; |
||||
|
||||
CodeMirror.simpleMode = function(config, states) { |
||||
ensureState(states, "start"); |
||||
var states_ = {}, meta = states.meta || {}, hasIndentation = false; |
||||
for (var state in states) if (state != meta && states.hasOwnProperty(state)) { |
||||
var list = states_[state] = [], orig = states[state]; |
||||
for (var i = 0; i < orig.length; i++) { |
||||
var data = orig[i]; |
||||
list.push(new Rule(data, states)); |
||||
if (data.indent || data.dedent) hasIndentation = true; |
||||
} |
||||
} |
||||
var mode = { |
||||
startState: function() { |
||||
return {state: "start", pending: null, |
||||
local: null, localState: null, |
||||
indent: hasIndentation ? [] : null}; |
||||
}, |
||||
copyState: function(state) { |
||||
var s = {state: state.state, pending: state.pending, |
||||
local: state.local, localState: null, |
||||
indent: state.indent && state.indent.slice(0)}; |
||||
if (state.localState) |
||||
s.localState = CodeMirror.copyState(state.local.mode, state.localState); |
||||
if (state.stack) |
||||
s.stack = state.stack.slice(0); |
||||
for (var pers = state.persistentStates; pers; pers = pers.next) |
||||
s.persistentStates = {mode: pers.mode, |
||||
spec: pers.spec, |
||||
state: pers.state == state.localState ? s.localState : CodeMirror.copyState(pers.mode, pers.state), |
||||
next: s.persistentStates}; |
||||
return s; |
||||
}, |
||||
token: tokenFunction(states_, config), |
||||
innerMode: function(state) { return state.local && {mode: state.local.mode, state: state.localState}; }, |
||||
indent: indentFunction(states_, meta) |
||||
}; |
||||
if (meta) for (var prop in meta) if (meta.hasOwnProperty(prop)) |
||||
mode[prop] = meta[prop]; |
||||
return mode; |
||||
}; |
||||
|
||||
function ensureState(states, name) { |
||||
if (!states.hasOwnProperty(name)) |
||||
throw new Error("Undefined state " + name + " in simple mode"); |
||||
} |
||||
|
||||
function toRegex(val, caret) { |
||||
if (!val) return /(?:)/; |
||||
var flags = ""; |
||||
if (val instanceof RegExp) { |
||||
if (val.ignoreCase) flags = "i"; |
||||
val = val.source; |
||||
} else { |
||||
val = String(val); |
||||
} |
||||
return new RegExp((caret === false ? "" : "^") + "(?:" + val + ")", flags); |
||||
} |
||||
|
||||
function asToken(val) { |
||||
if (!val) return null; |
||||
if (typeof val == "string") return val.replace(/\./g, " "); |
||||
var result = []; |
||||
for (var i = 0; i < val.length; i++) |
||||
result.push(val[i] && val[i].replace(/\./g, " ")); |
||||
return result; |
||||
} |
||||
|
||||
function Rule(data, states) { |
||||
if (data.next || data.push) ensureState(states, data.next || data.push); |
||||
this.regex = toRegex(data.regex); |
||||
this.token = asToken(data.token); |
||||
this.data = data; |
||||
} |
||||
|
||||
function tokenFunction(states, config) { |
||||
return function(stream, state) { |
||||
if (state.pending) { |
||||
var pend = state.pending.shift(); |
||||
if (state.pending.length == 0) state.pending = null; |
||||
stream.pos += pend.text.length; |
||||
return pend.token; |
||||
} |
||||
|
||||
if (state.local) { |
||||
if (state.local.end && stream.match(state.local.end)) { |
||||
var tok = state.local.endToken || null; |
||||
state.local = state.localState = null; |
||||
return tok; |
||||
} else { |
||||
var tok = state.local.mode.token(stream, state.localState), m; |
||||
if (state.local.endScan && (m = state.local.endScan.exec(stream.current()))) |
||||
stream.pos = stream.start + m.index; |
||||
return tok; |
||||
} |
||||
} |
||||
|
||||
var curState = states[state.state]; |
||||
for (var i = 0; i < curState.length; i++) { |
||||
var rule = curState[i]; |
||||
var matches = (!rule.data.sol || stream.sol()) && stream.match(rule.regex); |
||||
if (matches) { |
||||
if (rule.data.next) { |
||||
state.state = rule.data.next; |
||||
} else if (rule.data.push) { |
||||
(state.stack || (state.stack = [])).push(state.state); |
||||
state.state = rule.data.push; |
||||
} else if (rule.data.pop && state.stack && state.stack.length) { |
||||
state.state = state.stack.pop(); |
||||
} |
||||
|
||||
if (rule.data.mode) |
||||
enterLocalMode(config, state, rule.data.mode, rule.token); |
||||
if (rule.data.indent) |
||||
state.indent.push(stream.indentation() + config.indentUnit); |
||||
if (rule.data.dedent) |
||||
state.indent.pop(); |
||||
if (matches.length > 2) { |
||||
state.pending = []; |
||||
for (var j = 2; j < matches.length; j++) |
||||
if (matches[j]) |
||||
state.pending.push({text: matches[j], token: rule.token[j - 1]}); |
||||
stream.backUp(matches[0].length - (matches[1] ? matches[1].length : 0)); |
||||
return rule.token[0]; |
||||
} else if (rule.token && rule.token.join) { |
||||
return rule.token[0]; |
||||
} else { |
||||
return rule.token; |
||||
} |
||||
} |
||||
} |
||||
stream.next(); |
||||
return null; |
||||
}; |
||||
} |
||||
|
||||
function cmp(a, b) { |
||||
if (a === b) return true; |
||||
if (!a || typeof a != "object" || !b || typeof b != "object") return false; |
||||
var props = 0; |
||||
for (var prop in a) if (a.hasOwnProperty(prop)) { |
||||
if (!b.hasOwnProperty(prop) || !cmp(a[prop], b[prop])) return false; |
||||
props++; |
||||
} |
||||
for (var prop in b) if (b.hasOwnProperty(prop)) props--; |
||||
return props == 0; |
||||
} |
||||
|
||||
function enterLocalMode(config, state, spec, token) { |
||||
var pers; |
||||
if (spec.persistent) for (var p = state.persistentStates; p && !pers; p = p.next) |
||||
if (spec.spec ? cmp(spec.spec, p.spec) : spec.mode == p.mode) pers = p; |
||||
var mode = pers ? pers.mode : spec.mode || CodeMirror.getMode(config, spec.spec); |
||||
var lState = pers ? pers.state : CodeMirror.startState(mode); |
||||
if (spec.persistent && !pers) |
||||
state.persistentStates = {mode: mode, spec: spec.spec, state: lState, next: state.persistentStates}; |
||||
|
||||
state.localState = lState; |
||||
state.local = {mode: mode, |
||||
end: spec.end && toRegex(spec.end), |
||||
endScan: spec.end && spec.forceEnd !== false && toRegex(spec.end, false), |
||||
endToken: token && token.join ? token[token.length - 1] : token}; |
||||
} |
||||
|
||||
function indexOf(val, arr) { |
||||
for (var i = 0; i < arr.length; i++) if (arr[i] === val) return true; |
||||
} |
||||
|
||||
function indentFunction(states, meta) { |
||||
return function(state, textAfter, line) { |
||||
if (state.local && state.local.mode.indent) |
||||
return state.local.mode.indent(state.localState, textAfter, line); |
||||
if (state.indent == null || state.local || meta.dontIndentStates && indexOf(state.state, meta.dontIndentStates) > -1) |
||||
return CodeMirror.Pass; |
||||
|
||||
var pos = state.indent.length - 1, rules = states[state.state]; |
||||
scan: for (;;) { |
||||
for (var i = 0; i < rules.length; i++) { |
||||
var rule = rules[i]; |
||||
if (rule.data.dedent && rule.data.dedentIfLineStart !== false) { |
||||
var m = rule.regex.exec(textAfter); |
||||
if (m && m[0]) { |
||||
pos--; |
||||
if (rule.next || rule.push) rules = states[rule.next || rule.push]; |
||||
textAfter = textAfter.slice(m[0].length); |
||||
continue scan; |
||||
} |
||||
} |
||||
} |
||||
break; |
||||
} |
||||
return pos < 0 ? 0 : state.indent[pos]; |
||||
}; |
||||
} |
||||
}); |
@ -0,0 +1,174 @@ |
||||
// 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("apl", function() { |
||||
var builtInOps = { |
||||
".": "innerProduct", |
||||
"\\": "scan", |
||||
"/": "reduce", |
||||
"⌿": "reduce1Axis", |
||||
"⍀": "scan1Axis", |
||||
"¨": "each", |
||||
"⍣": "power" |
||||
}; |
||||
var builtInFuncs = { |
||||
"+": ["conjugate", "add"], |
||||
"−": ["negate", "subtract"], |
||||
"×": ["signOf", "multiply"], |
||||
"÷": ["reciprocal", "divide"], |
||||
"⌈": ["ceiling", "greaterOf"], |
||||
"⌊": ["floor", "lesserOf"], |
||||
"∣": ["absolute", "residue"], |
||||
"⍳": ["indexGenerate", "indexOf"], |
||||
"?": ["roll", "deal"], |
||||
"⋆": ["exponentiate", "toThePowerOf"], |
||||
"⍟": ["naturalLog", "logToTheBase"], |
||||
"○": ["piTimes", "circularFuncs"], |
||||
"!": ["factorial", "binomial"], |
||||
"⌹": ["matrixInverse", "matrixDivide"], |
||||
"<": [null, "lessThan"], |
||||
"≤": [null, "lessThanOrEqual"], |
||||
"=": [null, "equals"], |
||||
">": [null, "greaterThan"], |
||||
"≥": [null, "greaterThanOrEqual"], |
||||
"≠": [null, "notEqual"], |
||||
"≡": ["depth", "match"], |
||||
"≢": [null, "notMatch"], |
||||
"∈": ["enlist", "membership"], |
||||
"⍷": [null, "find"], |
||||
"∪": ["unique", "union"], |
||||
"∩": [null, "intersection"], |
||||
"∼": ["not", "without"], |
||||
"∨": [null, "or"], |
||||
"∧": [null, "and"], |
||||
"⍱": [null, "nor"], |
||||
"⍲": [null, "nand"], |
||||
"⍴": ["shapeOf", "reshape"], |
||||
",": ["ravel", "catenate"], |
||||
"⍪": [null, "firstAxisCatenate"], |
||||
"⌽": ["reverse", "rotate"], |
||||
"⊖": ["axis1Reverse", "axis1Rotate"], |
||||
"⍉": ["transpose", null], |
||||
"↑": ["first", "take"], |
||||
"↓": [null, "drop"], |
||||
"⊂": ["enclose", "partitionWithAxis"], |
||||
"⊃": ["diclose", "pick"], |
||||
"⌷": [null, "index"], |
||||
"⍋": ["gradeUp", null], |
||||
"⍒": ["gradeDown", null], |
||||
"⊤": ["encode", null], |
||||
"⊥": ["decode", null], |
||||
"⍕": ["format", "formatByExample"], |
||||
"⍎": ["execute", null], |
||||
"⊣": ["stop", "left"], |
||||
"⊢": ["pass", "right"] |
||||
}; |
||||
|
||||
var isOperator = /[\.\/⌿⍀¨⍣]/; |
||||
var isNiladic = /⍬/; |
||||
var isFunction = /[\+−×÷⌈⌊∣⍳\?⋆⍟○!⌹<≤=>≥≠≡≢∈⍷∪∩∼∨∧⍱⍲⍴,⍪⌽⊖⍉↑↓⊂⊃⌷⍋⍒⊤⊥⍕⍎⊣⊢]/; |
||||
var isArrow = /←/; |
||||
var isComment = /[⍝#].*$/; |
||||
|
||||
var stringEater = function(type) { |
||||
var prev; |
||||
prev = false; |
||||
return function(c) { |
||||
prev = c; |
||||
if (c === type) { |
||||
return prev === "\\"; |
||||
} |
||||
return true; |
||||
}; |
||||
}; |
||||
return { |
||||
startState: function() { |
||||
return { |
||||
prev: false, |
||||
func: false, |
||||
op: false, |
||||
string: false, |
||||
escape: false |
||||
}; |
||||
}, |
||||
token: function(stream, state) { |
||||
var ch, funcName; |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
ch = stream.next(); |
||||
if (ch === '"' || ch === "'") { |
||||
stream.eatWhile(stringEater(ch)); |
||||
stream.next(); |
||||
state.prev = true; |
||||
return "string"; |
||||
} |
||||
if (/[\[{\(]/.test(ch)) { |
||||
state.prev = false; |
||||
return null; |
||||
} |
||||
if (/[\]}\)]/.test(ch)) { |
||||
state.prev = true; |
||||
return null; |
||||
} |
||||
if (isNiladic.test(ch)) { |
||||
state.prev = false; |
||||
return "niladic"; |
||||
} |
||||
if (/[¯\d]/.test(ch)) { |
||||
if (state.func) { |
||||
state.func = false; |
||||
state.prev = false; |
||||
} else { |
||||
state.prev = true; |
||||
} |
||||
stream.eatWhile(/[\w\.]/); |
||||
return "number"; |
||||
} |
||||
if (isOperator.test(ch)) { |
||||
return "operator apl-" + builtInOps[ch]; |
||||
} |
||||
if (isArrow.test(ch)) { |
||||
return "apl-arrow"; |
||||
} |
||||
if (isFunction.test(ch)) { |
||||
funcName = "apl-"; |
||||
if (builtInFuncs[ch] != null) { |
||||
if (state.prev) { |
||||
funcName += builtInFuncs[ch][1]; |
||||
} else { |
||||
funcName += builtInFuncs[ch][0]; |
||||
} |
||||
} |
||||
state.func = true; |
||||
state.prev = false; |
||||
return "function " + funcName; |
||||
} |
||||
if (isComment.test(ch)) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
if (ch === "∘" && stream.peek() === ".") { |
||||
stream.next(); |
||||
return "function jot-dot"; |
||||
} |
||||
stream.eatWhile(/[\w\$_]/); |
||||
state.prev = true; |
||||
return "keyword"; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/apl", "apl"); |
||||
|
||||
}); |
@ -0,0 +1,72 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: APL mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="./apl.js"></script> |
||||
<style> |
||||
.CodeMirror { border: 2px inset #dee; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">APL</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>APL mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
⍝ Conway's game of life |
||||
|
||||
⍝ This example was inspired by the impressive demo at |
||||
⍝ http://www.youtube.com/watch?v=a9xAKttWgP4 |
||||
|
||||
⍝ Create a matrix: |
||||
⍝ 0 1 1 |
||||
⍝ 1 1 0 |
||||
⍝ 0 1 0 |
||||
creature ← (3 3 ⍴ ⍳ 9) ∈ 1 2 3 4 7 ⍝ Original creature from demo |
||||
creature ← (3 3 ⍴ ⍳ 9) ∈ 1 3 6 7 8 ⍝ Glider |
||||
|
||||
⍝ Place the creature on a larger board, near the centre |
||||
board ← ¯1 ⊖ ¯2 ⌽ 5 7 ↑ creature |
||||
|
||||
⍝ A function to move from one generation to the next |
||||
life ← {∨/ 1 ⍵ ∧ 3 4 = ⊂+/ +⌿ 1 0 ¯1 ∘.⊖ 1 0 ¯1 ⌽¨ ⊂⍵} |
||||
|
||||
⍝ Compute n-th generation and format it as a |
||||
⍝ character matrix |
||||
gen ← {' #'[(life ⍣ ⍵) board]} |
||||
|
||||
⍝ Show first three generations |
||||
(gen 1) (gen 2) (gen 3) |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/apl" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Simple mode that tries to handle APL as well as it can.</p> |
||||
<p>It attempts to label functions/operators based upon |
||||
monadic/dyadic usage (but this is far from fully fleshed out). |
||||
This means there are meaningful classnames so hover states can |
||||
have popups etc.</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/apl</code> (APL code)</p> |
||||
</article> |
@ -0,0 +1,73 @@ |
||||
// 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"; |
||||
|
||||
function errorIfNotEmpty(stream) { |
||||
var nonWS = stream.match(/^\s*\S/); |
||||
stream.skipToEnd(); |
||||
return nonWS ? "error" : null; |
||||
} |
||||
|
||||
CodeMirror.defineMode("asciiarmor", function() { |
||||
return { |
||||
token: function(stream, state) { |
||||
var m; |
||||
if (state.state == "top") { |
||||
if (stream.sol() && (m = stream.match(/^-----BEGIN (.*)?-----\s*$/))) { |
||||
state.state = "headers"; |
||||
state.type = m[1]; |
||||
return "tag"; |
||||
} |
||||
return errorIfNotEmpty(stream); |
||||
} else if (state.state == "headers") { |
||||
if (stream.sol() && stream.match(/^\w+:/)) { |
||||
state.state = "header"; |
||||
return "atom"; |
||||
} else { |
||||
var result = errorIfNotEmpty(stream); |
||||
if (result) state.state = "body"; |
||||
return result; |
||||
} |
||||
} else if (state.state == "header") { |
||||
stream.skipToEnd(); |
||||
state.state = "headers"; |
||||
return "string"; |
||||
} else if (state.state == "body") { |
||||
if (stream.sol() && (m = stream.match(/^-----END (.*)?-----\s*$/))) { |
||||
if (m[1] != state.type) return "error"; |
||||
state.state = "end"; |
||||
return "tag"; |
||||
} else { |
||||
if (stream.eatWhile(/[A-Za-z0-9+\/=]/)) { |
||||
return null; |
||||
} else { |
||||
stream.next(); |
||||
return "error"; |
||||
} |
||||
} |
||||
} else if (state.state == "end") { |
||||
return errorIfNotEmpty(stream); |
||||
} |
||||
}, |
||||
blankLine: function(state) { |
||||
if (state.state == "headers") state.state = "body"; |
||||
}, |
||||
startState: function() { |
||||
return {state: "top", type: null}; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("application/pgp", "asciiarmor"); |
||||
CodeMirror.defineMIME("application/pgp-keys", "asciiarmor"); |
||||
CodeMirror.defineMIME("application/pgp-signature", "asciiarmor"); |
||||
}); |
@ -0,0 +1,46 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: ASCII Armor (PGP) mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="asciiarmor.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">ASCII Armor</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>ASCII Armor (PGP) mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
-----BEGIN PGP MESSAGE----- |
||||
Version: OpenPrivacy 0.99 |
||||
|
||||
yDgBO22WxBHv7O8X7O/jygAEzol56iUKiXmV+XmpCtmpqQUKiQrFqclFqUDBovzS |
||||
vBSFjNSiVHsuAA== |
||||
=njUN |
||||
-----END PGP MESSAGE----- |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types |
||||
defined:</strong> <code>application/pgp</code>, <code>application/pgp-keys</code>, <code>application/pgp-signature</code></p> |
||||
|
||||
</article> |
@ -0,0 +1,204 @@ |
||||
// 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("asn.1", function(config, parserConfig) { |
||||
var indentUnit = config.indentUnit, |
||||
keywords = parserConfig.keywords || {}, |
||||
cmipVerbs = parserConfig.cmipVerbs || {}, |
||||
compareTypes = parserConfig.compareTypes || {}, |
||||
status = parserConfig.status || {}, |
||||
tags = parserConfig.tags || {}, |
||||
storage = parserConfig.storage || {}, |
||||
modifier = parserConfig.modifier || {}, |
||||
accessTypes = parserConfig.accessTypes|| {}, |
||||
multiLineStrings = parserConfig.multiLineStrings, |
||||
indentStatements = parserConfig.indentStatements !== false; |
||||
var isOperatorChar = /[\|\^]/; |
||||
var curPunc; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (ch == '"' || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (/[\[\]\(\){}:=,;]/.test(ch)) { |
||||
curPunc = ch; |
||||
return "punctuation"; |
||||
} |
||||
if (ch == "-"){ |
||||
if (stream.eat("-")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (/\d/.test(ch)) { |
||||
stream.eatWhile(/[\w\.]/); |
||||
return "number"; |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
|
||||
stream.eatWhile(/[\w\-]/); |
||||
var cur = stream.current(); |
||||
if (keywords.propertyIsEnumerable(cur)) return "keyword"; |
||||
if (cmipVerbs.propertyIsEnumerable(cur)) return "variable cmipVerbs"; |
||||
if (compareTypes.propertyIsEnumerable(cur)) return "atom compareTypes"; |
||||
if (status.propertyIsEnumerable(cur)) return "comment status"; |
||||
if (tags.propertyIsEnumerable(cur)) return "variable-3 tags"; |
||||
if (storage.propertyIsEnumerable(cur)) return "builtin storage"; |
||||
if (modifier.propertyIsEnumerable(cur)) return "string-2 modifier"; |
||||
if (accessTypes.propertyIsEnumerable(cur)) return "atom accessTypes"; |
||||
|
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped){ |
||||
var afterNext = stream.peek(); |
||||
//look if the character if the quote is like the B in '10100010'B
|
||||
if (afterNext){ |
||||
afterNext = afterNext.toLowerCase(); |
||||
if(afterNext == "b" || afterNext == "h" || afterNext == "o") |
||||
stream.next(); |
||||
} |
||||
end = true; break; |
||||
} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !(escaped || multiLineStrings)) |
||||
state.tokenize = null; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
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) { |
||||
var indent = state.indented; |
||||
if (state.context && state.context.type == "statement") |
||||
indent = state.context.indented; |
||||
return state.context = new Context(indent, 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: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), |
||||
indented: 0, |
||||
startOfLine: true |
||||
}; |
||||
}, |
||||
|
||||
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; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
curPunc = null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
if ((curPunc == ";" || curPunc == ":" || curPunc == ",") |
||||
&& ctx.type == "statement"){ |
||||
popContext(state); |
||||
} |
||||
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 (indentStatements && (((ctx.type == "}" || ctx.type == "top") |
||||
&& curPunc != ';') || (ctx.type == "statement" |
||||
&& curPunc == "newstatement"))) |
||||
pushContext(state, stream.column(), "statement"); |
||||
|
||||
state.startOfLine = false; |
||||
return style; |
||||
}, |
||||
|
||||
electricChars: "{}", |
||||
lineComment: "--", |
||||
fold: "brace" |
||||
}; |
||||
}); |
||||
|
||||
function words(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
|
||||
CodeMirror.defineMIME("text/x-ttcn-asn", { |
||||
name: "asn.1", |
||||
keywords: words("DEFINITIONS OBJECTS IF DERIVED INFORMATION ACTION" + |
||||
" REPLY ANY NAMED CHARACTERIZED BEHAVIOUR REGISTERED" + |
||||
" WITH AS IDENTIFIED CONSTRAINED BY PRESENT BEGIN" + |
||||
" IMPORTS FROM UNITS SYNTAX MIN-ACCESS MAX-ACCESS" + |
||||
" MINACCESS MAXACCESS REVISION STATUS DESCRIPTION" + |
||||
" SEQUENCE SET COMPONENTS OF CHOICE DistinguishedName" + |
||||
" ENUMERATED SIZE MODULE END INDEX AUGMENTS EXTENSIBILITY" + |
||||
" IMPLIED EXPORTS"), |
||||
cmipVerbs: words("ACTIONS ADD GET NOTIFICATIONS REPLACE REMOVE"), |
||||
compareTypes: words("OPTIONAL DEFAULT MANAGED MODULE-TYPE MODULE_IDENTITY" + |
||||
" MODULE-COMPLIANCE OBJECT-TYPE OBJECT-IDENTITY" + |
||||
" OBJECT-COMPLIANCE MODE CONFIRMED CONDITIONAL" + |
||||
" SUBORDINATE SUPERIOR CLASS TRUE FALSE NULL" + |
||||
" TEXTUAL-CONVENTION"), |
||||
status: words("current deprecated mandatory obsolete"), |
||||
tags: words("APPLICATION AUTOMATIC EXPLICIT IMPLICIT PRIVATE TAGS" + |
||||
" UNIVERSAL"), |
||||
storage: words("BOOLEAN INTEGER OBJECT IDENTIFIER BIT OCTET STRING" + |
||||
" UTCTime InterfaceIndex IANAifType CMIP-Attribute" + |
||||
" REAL PACKAGE PACKAGES IpAddress PhysAddress" + |
||||
" NetworkAddress BITS BMPString TimeStamp TimeTicks" + |
||||
" TruthValue RowStatus DisplayString GeneralString" + |
||||
" GraphicString IA5String NumericString" + |
||||
" PrintableString SnmpAdminAtring TeletexString" + |
||||
" UTF8String VideotexString VisibleString StringStore" + |
||||
" ISO646String T61String UniversalString Unsigned32" + |
||||
" Integer32 Gauge Gauge32 Counter Counter32 Counter64"), |
||||
modifier: words("ATTRIBUTE ATTRIBUTES MANDATORY-GROUP MANDATORY-GROUPS" + |
||||
" GROUP GROUPS ELEMENTS EQUALITY ORDERING SUBSTRINGS" + |
||||
" DEFINED"), |
||||
accessTypes: words("not-accessible accessible-for-notify read-only" + |
||||
" read-create read-write"), |
||||
multiLineStrings: true |
||||
}); |
||||
}); |
@ -0,0 +1,196 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/* |
||||
* ===================================================================================== |
||||
* |
||||
* Filename: mode/asterisk/asterisk.js |
||||
* |
||||
* Description: CodeMirror mode for Asterisk dialplan |
||||
* |
||||
* Created: 05/17/2012 09:20:25 PM |
||||
* Revision: none |
||||
* |
||||
* Author: Stas Kobzar (stas@modulis.ca), |
||||
* Company: Modulis.ca Inc. |
||||
* |
||||
* ===================================================================================== |
||||
*/ |
||||
|
||||
(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("asterisk", function() { |
||||
var atoms = ["exten", "same", "include","ignorepat","switch"], |
||||
dpcmd = ["#include","#exec"], |
||||
apps = [ |
||||
"addqueuemember","adsiprog","aelsub","agentlogin","agentmonitoroutgoing","agi", |
||||
"alarmreceiver","amd","answer","authenticate","background","backgrounddetect", |
||||
"bridge","busy","callcompletioncancel","callcompletionrequest","celgenuserevent", |
||||
"changemonitor","chanisavail","channelredirect","chanspy","clearhash","confbridge", |
||||
"congestion","continuewhile","controlplayback","dahdiacceptr2call","dahdibarge", |
||||
"dahdiras","dahdiscan","dahdisendcallreroutingfacility","dahdisendkeypadfacility", |
||||
"datetime","dbdel","dbdeltree","deadagi","dial","dictate","directory","disa", |
||||
"dumpchan","eagi","echo","endwhile","exec","execif","execiftime","exitwhile","extenspy", |
||||
"externalivr","festival","flash","followme","forkcdr","getcpeid","gosub","gosubif", |
||||
"goto","gotoif","gotoiftime","hangup","iax2provision","ices","importvar","incomplete", |
||||
"ivrdemo","jabberjoin","jabberleave","jabbersend","jabbersendgroup","jabberstatus", |
||||
"jack","log","macro","macroexclusive","macroexit","macroif","mailboxexists","meetme", |
||||
"meetmeadmin","meetmechanneladmin","meetmecount","milliwatt","minivmaccmess","minivmdelete", |
||||
"minivmgreet","minivmmwi","minivmnotify","minivmrecord","mixmonitor","monitor","morsecode", |
||||
"mp3player","mset","musiconhold","nbscat","nocdr","noop","odbc","odbc","odbcfinish", |
||||
"originate","ospauth","ospfinish","osplookup","ospnext","page","park","parkandannounce", |
||||
"parkedcall","pausemonitor","pausequeuemember","pickup","pickupchan","playback","playtones", |
||||
"privacymanager","proceeding","progress","queue","queuelog","raiseexception","read","readexten", |
||||
"readfile","receivefax","receivefax","receivefax","record","removequeuemember", |
||||
"resetcdr","retrydial","return","ringing","sayalpha","saycountedadj","saycountednoun", |
||||
"saycountpl","saydigits","saynumber","sayphonetic","sayunixtime","senddtmf","sendfax", |
||||
"sendfax","sendfax","sendimage","sendtext","sendurl","set","setamaflags", |
||||
"setcallerpres","setmusiconhold","sipaddheader","sipdtmfmode","sipremoveheader","skel", |
||||
"slastation","slatrunk","sms","softhangup","speechactivategrammar","speechbackground", |
||||
"speechcreate","speechdeactivategrammar","speechdestroy","speechloadgrammar","speechprocessingsound", |
||||
"speechstart","speechunloadgrammar","stackpop","startmusiconhold","stopmixmonitor","stopmonitor", |
||||
"stopmusiconhold","stopplaytones","system","testclient","testserver","transfer","tryexec", |
||||
"trysystem","unpausemonitor","unpausequeuemember","userevent","verbose","vmauthenticate", |
||||
"vmsayname","voicemail","voicemailmain","wait","waitexten","waitfornoise","waitforring", |
||||
"waitforsilence","waitmusiconhold","waituntil","while","zapateller" |
||||
]; |
||||
|
||||
function basicToken(stream,state){ |
||||
var cur = ''; |
||||
var ch = stream.next(); |
||||
// comment
|
||||
if(ch == ";") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
// context
|
||||
if(ch == '[') { |
||||
stream.skipTo(']'); |
||||
stream.eat(']'); |
||||
return "header"; |
||||
} |
||||
// string
|
||||
if(ch == '"') { |
||||
stream.skipTo('"'); |
||||
return "string"; |
||||
} |
||||
if(ch == "'") { |
||||
stream.skipTo("'"); |
||||
return "string-2"; |
||||
} |
||||
// dialplan commands
|
||||
if(ch == '#') { |
||||
stream.eatWhile(/\w/); |
||||
cur = stream.current(); |
||||
if(dpcmd.indexOf(cur) !== -1) { |
||||
stream.skipToEnd(); |
||||
return "strong"; |
||||
} |
||||
} |
||||
// application args
|
||||
if(ch == '$'){ |
||||
var ch1 = stream.peek(); |
||||
if(ch1 == '{'){ |
||||
stream.skipTo('}'); |
||||
stream.eat('}'); |
||||
return "variable-3"; |
||||
} |
||||
} |
||||
// extension
|
||||
stream.eatWhile(/\w/); |
||||
cur = stream.current(); |
||||
if(atoms.indexOf(cur) !== -1) { |
||||
state.extenStart = true; |
||||
switch(cur) { |
||||
case 'same': state.extenSame = true; break; |
||||
case 'include': |
||||
case 'switch': |
||||
case 'ignorepat': |
||||
state.extenInclude = true;break; |
||||
default:break; |
||||
} |
||||
return "atom"; |
||||
} |
||||
} |
||||
|
||||
return { |
||||
startState: function() { |
||||
return { |
||||
extenStart: false, |
||||
extenSame: false, |
||||
extenInclude: false, |
||||
extenExten: false, |
||||
extenPriority: false, |
||||
extenApplication: false |
||||
}; |
||||
}, |
||||
token: function(stream, state) { |
||||
|
||||
var cur = ''; |
||||
if(stream.eatSpace()) return null; |
||||
// extension started
|
||||
if(state.extenStart){ |
||||
stream.eatWhile(/[^\s]/); |
||||
cur = stream.current(); |
||||
if(/^=>?$/.test(cur)){ |
||||
state.extenExten = true; |
||||
state.extenStart = false; |
||||
return "strong"; |
||||
} else { |
||||
state.extenStart = false; |
||||
stream.skipToEnd(); |
||||
return "error"; |
||||
} |
||||
} else if(state.extenExten) { |
||||
// set exten and priority
|
||||
state.extenExten = false; |
||||
state.extenPriority = true; |
||||
stream.eatWhile(/[^,]/); |
||||
if(state.extenInclude) { |
||||
stream.skipToEnd(); |
||||
state.extenPriority = false; |
||||
state.extenInclude = false; |
||||
} |
||||
if(state.extenSame) { |
||||
state.extenPriority = false; |
||||
state.extenSame = false; |
||||
state.extenApplication = true; |
||||
} |
||||
return "tag"; |
||||
} else if(state.extenPriority) { |
||||
state.extenPriority = false; |
||||
state.extenApplication = true; |
||||
stream.next(); // get comma
|
||||
if(state.extenSame) return null; |
||||
stream.eatWhile(/[^,]/); |
||||
return "number"; |
||||
} else if(state.extenApplication) { |
||||
stream.eatWhile(/,/); |
||||
cur = stream.current(); |
||||
if(cur === ',') return null; |
||||
stream.eatWhile(/\w/); |
||||
cur = stream.current().toLowerCase(); |
||||
state.extenApplication = false; |
||||
if(apps.indexOf(cur) !== -1){ |
||||
return "def strong"; |
||||
} |
||||
} else{ |
||||
return basicToken(stream,state); |
||||
} |
||||
|
||||
return null; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-asterisk", "asterisk"); |
||||
|
||||
}); |
@ -0,0 +1,154 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Asterisk dialplan mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="asterisk.js"></script> |
||||
<style> |
||||
.CodeMirror {border: 1px solid #999;} |
||||
.cm-s-default span.cm-arrow { color: red; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Asterisk dialplan</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Asterisk dialplan mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
; extensions.conf - the Asterisk dial plan |
||||
; |
||||
|
||||
[general] |
||||
; |
||||
; If static is set to no, or omitted, then the pbx_config will rewrite |
||||
; this file when extensions are modified. Remember that all comments |
||||
; made in the file will be lost when that happens. |
||||
static=yes |
||||
|
||||
#include "/etc/asterisk/additional_general.conf |
||||
|
||||
[iaxprovider] |
||||
switch => IAX2/user:[key]@myserver/mycontext |
||||
|
||||
[dynamic] |
||||
#exec /usr/bin/dynamic-peers.pl |
||||
|
||||
[trunkint] |
||||
; |
||||
; International long distance through trunk |
||||
; |
||||
exten => _9011.,1,Macro(dundi-e164,${EXTEN:4}) |
||||
exten => _9011.,n,Dial(${GLOBAL(TRUNK)}/${FILTER(0-9,${EXTEN:${GLOBAL(TRUNKMSD)}})}) |
||||
|
||||
[local] |
||||
; |
||||
; Master context for local, toll-free, and iaxtel calls only |
||||
; |
||||
ignorepat => 9 |
||||
include => default |
||||
|
||||
[demo] |
||||
include => stdexten |
||||
; |
||||
; We start with what to do when a call first comes in. |
||||
; |
||||
exten => s,1,Wait(1) ; Wait a second, just for fun |
||||
same => n,Answer ; Answer the line |
||||
same => n,Set(TIMEOUT(digit)=5) ; Set Digit Timeout to 5 seconds |
||||
same => n,Set(TIMEOUT(response)=10) ; Set Response Timeout to 10 seconds |
||||
same => n(restart),BackGround(demo-congrats) ; Play a congratulatory message |
||||
same => n(instruct),BackGround(demo-instruct) ; Play some instructions |
||||
same => n,WaitExten ; Wait for an extension to be dialed. |
||||
|
||||
exten => 2,1,BackGround(demo-moreinfo) ; Give some more information. |
||||
exten => 2,n,Goto(s,instruct) |
||||
|
||||
exten => 3,1,Set(LANGUAGE()=fr) ; Set language to french |
||||
exten => 3,n,Goto(s,restart) ; Start with the congratulations |
||||
|
||||
exten => 1000,1,Goto(default,s,1) |
||||
; |
||||
; We also create an example user, 1234, who is on the console and has |
||||
; voicemail, etc. |
||||
; |
||||
exten => 1234,1,Playback(transfer,skip) ; "Please hold while..." |
||||
; (but skip if channel is not up) |
||||
exten => 1234,n,Gosub(${EXTEN},stdexten(${GLOBAL(CONSOLE)})) |
||||
exten => 1234,n,Goto(default,s,1) ; exited Voicemail |
||||
|
||||
exten => 1235,1,Voicemail(1234,u) ; Right to voicemail |
||||
|
||||
exten => 1236,1,Dial(Console/dsp) ; Ring forever |
||||
exten => 1236,n,Voicemail(1234,b) ; Unless busy |
||||
|
||||
; |
||||
; # for when they're done with the demo |
||||
; |
||||
exten => #,1,Playback(demo-thanks) ; "Thanks for trying the demo" |
||||
exten => #,n,Hangup ; Hang them up. |
||||
|
||||
; |
||||
; A timeout and "invalid extension rule" |
||||
; |
||||
exten => t,1,Goto(#,1) ; If they take too long, give up |
||||
exten => i,1,Playback(invalid) ; "That's not valid, try again" |
||||
|
||||
; |
||||
; Create an extension, 500, for dialing the |
||||
; Asterisk demo. |
||||
; |
||||
exten => 500,1,Playback(demo-abouttotry); Let them know what's going on |
||||
exten => 500,n,Dial(IAX2/guest@pbx.digium.com/s@default) ; Call the Asterisk demo |
||||
exten => 500,n,Playback(demo-nogo) ; Couldn't connect to the demo site |
||||
exten => 500,n,Goto(s,6) ; Return to the start over message. |
||||
|
||||
; |
||||
; Create an extension, 600, for evaluating echo latency. |
||||
; |
||||
exten => 600,1,Playback(demo-echotest) ; Let them know what's going on |
||||
exten => 600,n,Echo ; Do the echo test |
||||
exten => 600,n,Playback(demo-echodone) ; Let them know it's over |
||||
exten => 600,n,Goto(s,6) ; Start over |
||||
|
||||
; |
||||
; You can use the Macro Page to intercom a individual user |
||||
exten => 76245,1,Macro(page,SIP/Grandstream1) |
||||
; or if your peernames are the same as extensions |
||||
exten => _7XXX,1,Macro(page,SIP/${EXTEN}) |
||||
; |
||||
; |
||||
; System Wide Page at extension 7999 |
||||
; |
||||
exten => 7999,1,Set(TIMEOUT(absolute)=60) |
||||
exten => 7999,2,Page(Local/Grandstream1@page&Local/Xlite1@page&Local/1234@page/n,d) |
||||
|
||||
; Give voicemail at extension 8500 |
||||
; |
||||
exten => 8500,1,VoicemailMain |
||||
exten => 8500,n,Goto(s,6) |
||||
|
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: "text/x-asterisk", |
||||
matchBrackets: true, |
||||
lineNumber: true |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-asterisk</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,85 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
// Brainfuck 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" |
||||
var reserve = "><+-.,[]".split(""); |
||||
/* |
||||
comments can be either: |
||||
placed behind lines |
||||
|
||||
+++ this is a comment |
||||
|
||||
where reserved characters cannot be used |
||||
or in a loop |
||||
[ |
||||
this is ok to use [ ] and stuff |
||||
] |
||||
or preceded by # |
||||
*/ |
||||
CodeMirror.defineMode("brainfuck", function() { |
||||
return { |
||||
startState: function() { |
||||
return { |
||||
commentLine: false, |
||||
left: 0, |
||||
right: 0, |
||||
commentLoop: false |
||||
} |
||||
}, |
||||
token: function(stream, state) { |
||||
if (stream.eatSpace()) return null |
||||
if(stream.sol()){ |
||||
state.commentLine = false; |
||||
} |
||||
var ch = stream.next().toString(); |
||||
if(reserve.indexOf(ch) !== -1){ |
||||
if(state.commentLine === true){ |
||||
if(stream.eol()){ |
||||
state.commentLine = false; |
||||
} |
||||
return "comment"; |
||||
} |
||||
if(ch === "]" || ch === "["){ |
||||
if(ch === "["){ |
||||
state.left++; |
||||
} |
||||
else{ |
||||
state.right++; |
||||
} |
||||
return "bracket"; |
||||
} |
||||
else if(ch === "+" || ch === "-"){ |
||||
return "keyword"; |
||||
} |
||||
else if(ch === "<" || ch === ">"){ |
||||
return "atom"; |
||||
} |
||||
else if(ch === "." || ch === ","){ |
||||
return "def"; |
||||
} |
||||
} |
||||
else{ |
||||
state.commentLine = true; |
||||
if(stream.eol()){ |
||||
state.commentLine = false; |
||||
} |
||||
return "comment"; |
||||
} |
||||
if(stream.eol()){ |
||||
state.commentLine = false; |
||||
} |
||||
} |
||||
}; |
||||
}); |
||||
CodeMirror.defineMIME("text/x-brainfuck","brainfuck") |
||||
}); |
@ -0,0 +1,85 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Brainfuck mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="./brainfuck.js"></script> |
||||
<style> |
||||
.CodeMirror { border: 2px inset #dee; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#"></a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Brainfuck mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
[ This program prints "Hello World!" and a newline to the screen, its |
||||
length is 106 active command characters [it is not the shortest.] |
||||
|
||||
This loop is a "comment loop", it's a simple way of adding a comment |
||||
to a BF program such that you don't have to worry about any command |
||||
characters. Any ".", ",", "+", "-", "<" and ">" characters are simply |
||||
ignored, the "[" and "]" characters just have to be balanced. |
||||
] |
||||
+++++ +++ Set Cell #0 to 8 |
||||
[ |
||||
>++++ Add 4 to Cell #1; this will always set Cell #1 to 4 |
||||
[ as the cell will be cleared by the loop |
||||
>++ Add 2 to Cell #2 |
||||
>+++ Add 3 to Cell #3 |
||||
>+++ Add 3 to Cell #4 |
||||
>+ Add 1 to Cell #5 |
||||
<<<<- Decrement the loop counter in Cell #1 |
||||
] Loop till Cell #1 is zero; number of iterations is 4 |
||||
>+ Add 1 to Cell #2 |
||||
>+ Add 1 to Cell #3 |
||||
>- Subtract 1 from Cell #4 |
||||
>>+ Add 1 to Cell #6 |
||||
[<] Move back to the first zero cell you find; this will |
||||
be Cell #1 which was cleared by the previous loop |
||||
<- Decrement the loop Counter in Cell #0 |
||||
] Loop till Cell #0 is zero; number of iterations is 8 |
||||
|
||||
The result of this is: |
||||
Cell No : 0 1 2 3 4 5 6 |
||||
Contents: 0 0 72 104 88 32 8 |
||||
Pointer : ^ |
||||
|
||||
>>. Cell #2 has value 72 which is 'H' |
||||
>---. Subtract 3 from Cell #3 to get 101 which is 'e' |
||||
+++++++..+++. Likewise for 'llo' from Cell #3 |
||||
>>. Cell #5 is 32 for the space |
||||
<-. Subtract 1 from Cell #4 for 87 to give a 'W' |
||||
<. Cell #3 was set to 'o' from the end of 'Hello' |
||||
+++.------.--------. Cell #3 for 'rl' and 'd' |
||||
>>+. Add 1 to Cell #5 gives us an exclamation point |
||||
>++. And finally a newline from Cell #6 |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-brainfuck" |
||||
}); |
||||
</script> |
||||
|
||||
<p>A mode for Brainfuck</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-brainfuck</code></p> |
||||
</article> |
@ -0,0 +1,786 @@ |
||||
// 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"; |
||||
|
||||
function Context(indented, column, type, info, align, prev) { |
||||
this.indented = indented; |
||||
this.column = column; |
||||
this.type = type; |
||||
this.info = info; |
||||
this.align = align; |
||||
this.prev = prev; |
||||
} |
||||
function pushContext(state, col, type, info) { |
||||
var indent = state.indented; |
||||
if (state.context && state.context.type != "statement" && type != "statement") |
||||
indent = state.context.indented; |
||||
return state.context = new Context(indent, col, type, info, 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; |
||||
} |
||||
|
||||
function typeBefore(stream, state, pos) { |
||||
if (state.prevToken == "variable" || state.prevToken == "variable-3") return true; |
||||
if (/\S(?:[^- ]>|[*\]])\s*$|\*$/.test(stream.string.slice(0, pos))) return true; |
||||
if (state.typeAtEndOfLine && stream.column() == stream.indentation()) return true; |
||||
} |
||||
|
||||
function isTopScope(context) { |
||||
for (;;) { |
||||
if (!context || context.type == "top") return true; |
||||
if (context.type == "}" && context.prev.info != "namespace") return false; |
||||
context = context.prev; |
||||
} |
||||
} |
||||
|
||||
CodeMirror.defineMode("clike", function(config, parserConfig) { |
||||
var indentUnit = config.indentUnit, |
||||
statementIndentUnit = parserConfig.statementIndentUnit || indentUnit, |
||||
dontAlignCalls = parserConfig.dontAlignCalls, |
||||
keywords = parserConfig.keywords || {}, |
||||
types = parserConfig.types || {}, |
||||
builtin = parserConfig.builtin || {}, |
||||
blockKeywords = parserConfig.blockKeywords || {}, |
||||
defKeywords = parserConfig.defKeywords || {}, |
||||
atoms = parserConfig.atoms || {}, |
||||
hooks = parserConfig.hooks || {}, |
||||
multiLineStrings = parserConfig.multiLineStrings, |
||||
indentStatements = parserConfig.indentStatements !== false, |
||||
indentSwitch = parserConfig.indentSwitch !== false, |
||||
namespaceSeparator = parserConfig.namespaceSeparator, |
||||
isPunctuationChar = parserConfig.isPunctuationChar || /[\[\]{}\(\),;\:\.]/, |
||||
numberStart = parserConfig.numberStart || /[\d\.]/, |
||||
number = parserConfig.number || /^(?:0x[a-f\d]+|0b[01]+|(?:\d+\.?\d*|\.\d+)(?:e[-+]?\d+)?)(u|ll?|l|f)?/i, |
||||
isOperatorChar = parserConfig.isOperatorChar || /[+\-*&%=<>!?|\/]/, |
||||
endStatement = parserConfig.endStatement || /^[;:,]$/; |
||||
|
||||
var curPunc, isDefKeyword; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (hooks[ch]) { |
||||
var result = hooks[ch](stream, state); |
||||
if (result !== false) return result; |
||||
} |
||||
if (ch == '"' || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (isPunctuationChar.test(ch)) { |
||||
curPunc = ch; |
||||
return null; |
||||
} |
||||
if (numberStart.test(ch)) { |
||||
stream.backUp(1) |
||||
if (stream.match(number)) return "number" |
||||
stream.next() |
||||
} |
||||
if (ch == "/") { |
||||
if (stream.eat("*")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenComment(stream, state); |
||||
} |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
while (!stream.match(/^\/[\/*]/, false) && stream.eat(isOperatorChar)) {} |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
if (namespaceSeparator) while (stream.match(namespaceSeparator)) |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
|
||||
var cur = stream.current(); |
||||
if (contains(keywords, cur)) { |
||||
if (contains(blockKeywords, cur)) curPunc = "newstatement"; |
||||
if (contains(defKeywords, cur)) isDefKeyword = true; |
||||
return "keyword"; |
||||
} |
||||
if (contains(types, cur)) return "variable-3"; |
||||
if (contains(builtin, cur)) { |
||||
if (contains(blockKeywords, cur)) curPunc = "newstatement"; |
||||
return "builtin"; |
||||
} |
||||
if (contains(atoms, cur)) return "atom"; |
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) {end = true; break;} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !(escaped || multiLineStrings)) |
||||
state.tokenize = null; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
function maybeEOL(stream, state) { |
||||
if (parserConfig.typeFirstDefinitions && stream.eol() && isTopScope(state.context)) |
||||
state.typeAtEndOfLine = typeBefore(stream, state, stream.pos) |
||||
} |
||||
|
||||
// Interface
|
||||
|
||||
return { |
||||
startState: function(basecolumn) { |
||||
return { |
||||
tokenize: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", null, false), |
||||
indented: 0, |
||||
startOfLine: true, |
||||
prevToken: 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; |
||||
} |
||||
if (stream.eatSpace()) { maybeEOL(stream, state); return null; } |
||||
curPunc = isDefKeyword = null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment" || style == "meta") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
if (endStatement.test(curPunc)) while (state.context.type == "statement") popContext(state); |
||||
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 (indentStatements && |
||||
(((ctx.type == "}" || ctx.type == "top") && curPunc != ";") || |
||||
(ctx.type == "statement" && curPunc == "newstatement"))) { |
||||
pushContext(state, stream.column(), "statement", stream.current()); |
||||
} |
||||
|
||||
if (style == "variable" && |
||||
((state.prevToken == "def" || |
||||
(parserConfig.typeFirstDefinitions && typeBefore(stream, state, stream.start) && |
||||
isTopScope(state.context) && stream.match(/^\s*\(/, false))))) |
||||
style = "def"; |
||||
|
||||
if (hooks.token) { |
||||
var result = hooks.token(stream, state, style); |
||||
if (result !== undefined) style = result; |
||||
} |
||||
|
||||
if (style == "def" && parserConfig.styleDefs === false) style = "variable"; |
||||
|
||||
state.startOfLine = false; |
||||
state.prevToken = isDefKeyword ? "def" : style || curPunc; |
||||
maybeEOL(stream, state); |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != tokenBase && state.tokenize != null || state.typeAtEndOfLine) return CodeMirror.Pass; |
||||
var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); |
||||
if (ctx.type == "statement" && firstChar == "}") ctx = ctx.prev; |
||||
if (parserConfig.dontIndentStatements) |
||||
while (ctx.type == "statement" && parserConfig.dontIndentStatements.test(ctx.info)) |
||||
ctx = ctx.prev |
||||
if (hooks.indent) { |
||||
var hook = hooks.indent(state, ctx, textAfter); |
||||
if (typeof hook == "number") return hook |
||||
} |
||||
var closing = firstChar == ctx.type; |
||||
var switchBlock = ctx.prev && ctx.prev.info == "switch"; |
||||
if (parserConfig.allmanIndentation && /[{(]/.test(firstChar)) { |
||||
while (ctx.type != "top" && ctx.type != "}") ctx = ctx.prev |
||||
return ctx.indented |
||||
} |
||||
if (ctx.type == "statement") |
||||
return ctx.indented + (firstChar == "{" ? 0 : statementIndentUnit); |
||||
if (ctx.align && (!dontAlignCalls || ctx.type != ")")) |
||||
return ctx.column + (closing ? 0 : 1); |
||||
if (ctx.type == ")" && !closing) |
||||
return ctx.indented + statementIndentUnit; |
||||
|
||||
return ctx.indented + (closing ? 0 : indentUnit) + |
||||
(!closing && switchBlock && !/^(?:case|default)\b/.test(textAfter) ? indentUnit : 0); |
||||
}, |
||||
|
||||
electricInput: indentSwitch ? /^\s*(?:case .*?:|default:|\{\}?|\})$/ : /^\s*[{}]$/, |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/", |
||||
lineComment: "//", |
||||
fold: "brace" |
||||
}; |
||||
}); |
||||
|
||||
function words(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
function contains(words, word) { |
||||
if (typeof words === "function") { |
||||
return words(word); |
||||
} else { |
||||
return words.propertyIsEnumerable(word); |
||||
} |
||||
} |
||||
var cKeywords = "auto if break case register continue return default do sizeof " + |
||||
"static else struct switch extern typedef union for goto while enum const volatile"; |
||||
var cTypes = "int long char short double float unsigned signed void size_t ptrdiff_t"; |
||||
|
||||
function cppHook(stream, state) { |
||||
if (!state.startOfLine) return false |
||||
for (var ch, next = null; ch = stream.peek();) { |
||||
if (ch == "\\" && stream.match(/^.$/)) { |
||||
next = cppHook |
||||
break |
||||
} else if (ch == "/" && stream.match(/^\/[\/\*]/, false)) { |
||||
break |
||||
} |
||||
stream.next() |
||||
} |
||||
state.tokenize = next |
||||
return "meta" |
||||
} |
||||
|
||||
function pointerHook(_stream, state) { |
||||
if (state.prevToken == "variable-3") return "variable-3"; |
||||
return false; |
||||
} |
||||
|
||||
function cpp14Literal(stream) { |
||||
stream.eatWhile(/[\w\.']/); |
||||
return "number"; |
||||
} |
||||
|
||||
function cpp11StringHook(stream, state) { |
||||
stream.backUp(1); |
||||
// Raw strings.
|
||||
if (stream.match(/(R|u8R|uR|UR|LR)/)) { |
||||
var match = stream.match(/"([^\s\\()]{0,16})\(/); |
||||
if (!match) { |
||||
return false; |
||||
} |
||||
state.cpp11RawStringDelim = match[1]; |
||||
state.tokenize = tokenRawString; |
||||
return tokenRawString(stream, state); |
||||
} |
||||
// Unicode strings/chars.
|
||||
if (stream.match(/(u8|u|U|L)/)) { |
||||
if (stream.match(/["']/, /* eat */ false)) { |
||||
return "string"; |
||||
} |
||||
return false; |
||||
} |
||||
// Ignore this hook.
|
||||
stream.next(); |
||||
return false; |
||||
} |
||||
|
||||
function cppLooksLikeConstructor(word) { |
||||
var lastTwo = /(\w+)::(\w+)$/.exec(word); |
||||
return lastTwo && lastTwo[1] == lastTwo[2]; |
||||
} |
||||
|
||||
// C#-style strings where "" escapes a quote.
|
||||
function tokenAtString(stream, state) { |
||||
var next; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == '"' && !stream.eat('"')) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
} |
||||
return "string"; |
||||
} |
||||
|
||||
// C++11 raw string literal is <prefix>"<delim>( anything )<delim>", where
|
||||
// <delim> can be a string up to 16 characters long.
|
||||
function tokenRawString(stream, state) { |
||||
// Escape characters that have special regex meanings.
|
||||
var delim = state.cpp11RawStringDelim.replace(/[^\w\s]/g, '\\$&'); |
||||
var match = stream.match(new RegExp(".*?\\)" + delim + '"')); |
||||
if (match) |
||||
state.tokenize = null; |
||||
else |
||||
stream.skipToEnd(); |
||||
return "string"; |
||||
} |
||||
|
||||
function def(mimes, mode) { |
||||
if (typeof mimes == "string") mimes = [mimes]; |
||||
var words = []; |
||||
function add(obj) { |
||||
if (obj) for (var prop in obj) if (obj.hasOwnProperty(prop)) |
||||
words.push(prop); |
||||
} |
||||
add(mode.keywords); |
||||
add(mode.types); |
||||
add(mode.builtin); |
||||
add(mode.atoms); |
||||
if (words.length) { |
||||
mode.helperType = mimes[0]; |
||||
CodeMirror.registerHelper("hintWords", mimes[0], words); |
||||
} |
||||
|
||||
for (var i = 0; i < mimes.length; ++i) |
||||
CodeMirror.defineMIME(mimes[i], mode); |
||||
} |
||||
|
||||
def(["text/x-csrc", "text/x-c", "text/x-chdr"], { |
||||
name: "clike", |
||||
keywords: words(cKeywords), |
||||
types: words(cTypes + " bool _Complex _Bool float_t double_t intptr_t intmax_t " + |
||||
"int8_t int16_t int32_t int64_t uintptr_t uintmax_t uint8_t uint16_t " + |
||||
"uint32_t uint64_t"), |
||||
blockKeywords: words("case do else for if switch while struct"), |
||||
defKeywords: words("struct"), |
||||
typeFirstDefinitions: true, |
||||
atoms: words("null true false"), |
||||
hooks: {"#": cppHook, "*": pointerHook}, |
||||
modeProps: {fold: ["brace", "include"]} |
||||
}); |
||||
|
||||
def(["text/x-c++src", "text/x-c++hdr"], { |
||||
name: "clike", |
||||
keywords: words(cKeywords + " asm dynamic_cast namespace reinterpret_cast try explicit new " + |
||||
"static_cast typeid catch operator template typename class friend private " + |
||||
"this using const_cast inline public throw virtual delete mutable protected " + |
||||
"alignas alignof constexpr decltype nullptr noexcept thread_local final " + |
||||
"static_assert override"), |
||||
types: words(cTypes + " bool wchar_t"), |
||||
blockKeywords: words("catch class do else finally for if struct switch try while"), |
||||
defKeywords: words("class namespace struct enum union"), |
||||
typeFirstDefinitions: true, |
||||
atoms: words("true false null"), |
||||
dontIndentStatements: /^template$/, |
||||
hooks: { |
||||
"#": cppHook, |
||||
"*": pointerHook, |
||||
"u": cpp11StringHook, |
||||
"U": cpp11StringHook, |
||||
"L": cpp11StringHook, |
||||
"R": cpp11StringHook, |
||||
"0": cpp14Literal, |
||||
"1": cpp14Literal, |
||||
"2": cpp14Literal, |
||||
"3": cpp14Literal, |
||||
"4": cpp14Literal, |
||||
"5": cpp14Literal, |
||||
"6": cpp14Literal, |
||||
"7": cpp14Literal, |
||||
"8": cpp14Literal, |
||||
"9": cpp14Literal, |
||||
token: function(stream, state, style) { |
||||
if (style == "variable" && stream.peek() == "(" && |
||||
(state.prevToken == ";" || state.prevToken == null || |
||||
state.prevToken == "}") && |
||||
cppLooksLikeConstructor(stream.current())) |
||||
return "def"; |
||||
} |
||||
}, |
||||
namespaceSeparator: "::", |
||||
modeProps: {fold: ["brace", "include"]} |
||||
}); |
||||
|
||||
def("text/x-java", { |
||||
name: "clike", |
||||
keywords: words("abstract assert break case catch class const continue default " + |
||||
"do else enum extends final finally float for goto if implements import " + |
||||
"instanceof interface native new package private protected public " + |
||||
"return static strictfp super switch synchronized this throw throws transient " + |
||||
"try volatile while"), |
||||
types: words("byte short int long float double boolean char void Boolean Byte Character Double Float " + |
||||
"Integer Long Number Object Short String StringBuffer StringBuilder Void"), |
||||
blockKeywords: words("catch class do else finally for if switch try while"), |
||||
defKeywords: words("class interface package enum"), |
||||
typeFirstDefinitions: true, |
||||
atoms: words("true false null"), |
||||
endStatement: /^[;:]$/, |
||||
number: /^(?:0x[a-f\d_]+|0b[01_]+|(?:[\d_]+\.?\d*|\.\d+)(?:e[-+]?[\d_]+)?)(u|ll?|l|f)?/i, |
||||
hooks: { |
||||
"@": function(stream) { |
||||
stream.eatWhile(/[\w\$_]/); |
||||
return "meta"; |
||||
} |
||||
}, |
||||
modeProps: {fold: ["brace", "import"]} |
||||
}); |
||||
|
||||
def("text/x-csharp", { |
||||
name: "clike", |
||||
keywords: words("abstract as async await base break case catch checked class const continue" + |
||||
" default delegate do else enum event explicit extern finally fixed for" + |
||||
" foreach goto if implicit in interface internal is lock namespace new" + |
||||
" operator out override params private protected public readonly ref return sealed" + |
||||
" sizeof stackalloc static struct switch this throw try typeof unchecked" + |
||||
" unsafe using virtual void volatile while add alias ascending descending dynamic from get" + |
||||
" global group into join let orderby partial remove select set value var yield"), |
||||
types: words("Action Boolean Byte Char DateTime DateTimeOffset Decimal Double Func" + |
||||
" Guid Int16 Int32 Int64 Object SByte Single String Task TimeSpan UInt16 UInt32" + |
||||
" UInt64 bool byte char decimal double short int long object" + |
||||
" sbyte float string ushort uint ulong"), |
||||
blockKeywords: words("catch class do else finally for foreach if struct switch try while"), |
||||
defKeywords: words("class interface namespace struct var"), |
||||
typeFirstDefinitions: true, |
||||
atoms: words("true false null"), |
||||
hooks: { |
||||
"@": function(stream, state) { |
||||
if (stream.eat('"')) { |
||||
state.tokenize = tokenAtString; |
||||
return tokenAtString(stream, state); |
||||
} |
||||
stream.eatWhile(/[\w\$_]/); |
||||
return "meta"; |
||||
} |
||||
} |
||||
}); |
||||
|
||||
function tokenTripleString(stream, state) { |
||||
var escaped = false; |
||||
while (!stream.eol()) { |
||||
if (!escaped && stream.match('"""')) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
escaped = stream.next() == "\\" && !escaped; |
||||
} |
||||
return "string"; |
||||
} |
||||
|
||||
def("text/x-scala", { |
||||
name: "clike", |
||||
keywords: words( |
||||
|
||||
/* scala */ |
||||
"abstract case catch class def do else extends final finally for forSome if " + |
||||
"implicit import lazy match new null object override package private protected return " + |
||||
"sealed super this throw trait try type val var while with yield _ : = => <- <: " + |
||||
"<% >: # @ " + |
||||
|
||||
/* package scala */ |
||||
"assert assume require print println printf readLine readBoolean readByte readShort " + |
||||
"readChar readInt readLong readFloat readDouble " + |
||||
|
||||
":: #:: " |
||||
), |
||||
types: words( |
||||
"AnyVal App Application Array BufferedIterator BigDecimal BigInt Char Console Either " + |
||||
"Enumeration Equiv Error Exception Fractional Function IndexedSeq Int Integral Iterable " + |
||||
"Iterator List Map Numeric Nil NotNull Option Ordered Ordering PartialFunction PartialOrdering " + |
||||
"Product Proxy Range Responder Seq Serializable Set Specializable Stream StringBuilder " + |
||||
"StringContext Symbol Throwable Traversable TraversableOnce Tuple Unit Vector " + |
||||
|
||||
/* package java.lang */ |
||||
"Boolean Byte Character CharSequence Class ClassLoader Cloneable Comparable " + |
||||
"Compiler Double Exception Float Integer Long Math Number Object Package Pair Process " + |
||||
"Runtime Runnable SecurityManager Short StackTraceElement StrictMath String " + |
||||
"StringBuffer System Thread ThreadGroup ThreadLocal Throwable Triple Void" |
||||
), |
||||
multiLineStrings: true, |
||||
blockKeywords: words("catch class do else finally for forSome if match switch try while"), |
||||
defKeywords: words("class def object package trait type val var"), |
||||
atoms: words("true false null"), |
||||
indentStatements: false, |
||||
indentSwitch: false, |
||||
hooks: { |
||||
"@": function(stream) { |
||||
stream.eatWhile(/[\w\$_]/); |
||||
return "meta"; |
||||
}, |
||||
'"': function(stream, state) { |
||||
if (!stream.match('""')) return false; |
||||
state.tokenize = tokenTripleString; |
||||
return state.tokenize(stream, state); |
||||
}, |
||||
"'": function(stream) { |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
return "atom"; |
||||
}, |
||||
"=": function(stream, state) { |
||||
var cx = state.context |
||||
if (cx.type == "}" && cx.align && stream.eat(">")) { |
||||
state.context = new Context(cx.indented, cx.column, cx.type, cx.info, null, cx.prev) |
||||
return "operator" |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
}, |
||||
modeProps: {closeBrackets: {triples: '"'}} |
||||
}); |
||||
|
||||
function tokenKotlinString(tripleString){ |
||||
return function (stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while (!stream.eol()) { |
||||
if (!tripleString && !escaped && stream.match('"') ) {end = true; break;} |
||||
if (tripleString && stream.match('"""')) {end = true; break;} |
||||
next = stream.next(); |
||||
if(!escaped && next == "$" && stream.match('{')) |
||||
stream.skipTo("}"); |
||||
escaped = !escaped && next == "\\" && !tripleString; |
||||
} |
||||
if (end || !tripleString) |
||||
state.tokenize = null; |
||||
return "string"; |
||||
} |
||||
} |
||||
|
||||
def("text/x-kotlin", { |
||||
name: "clike", |
||||
keywords: words( |
||||
/*keywords*/ |
||||
"package as typealias class interface this super val " + |
||||
"var fun for is in This throw return " + |
||||
"break continue object if else while do try when !in !is as? " + |
||||
|
||||
/*soft keywords*/ |
||||
"file import where by get set abstract enum open inner override private public internal " + |
||||
"protected catch finally out final vararg reified dynamic companion constructor init " + |
||||
"sealed field property receiver param sparam lateinit data inline noinline tailrec " + |
||||
"external annotation crossinline const operator infix" |
||||
), |
||||
types: words( |
||||
/* package java.lang */ |
||||
"Boolean Byte Character CharSequence Class ClassLoader Cloneable Comparable " + |
||||
"Compiler Double Exception Float Integer Long Math Number Object Package Pair Process " + |
||||
"Runtime Runnable SecurityManager Short StackTraceElement StrictMath String " + |
||||
"StringBuffer System Thread ThreadGroup ThreadLocal Throwable Triple Void" |
||||
), |
||||
intendSwitch: false, |
||||
indentStatements: false, |
||||
multiLineStrings: true, |
||||
blockKeywords: words("catch class do else finally for if where try while enum"), |
||||
defKeywords: words("class val var object package interface fun"), |
||||
atoms: words("true false null this"), |
||||
hooks: { |
||||
'"': function(stream, state) { |
||||
state.tokenize = tokenKotlinString(stream.match('""')); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
}, |
||||
modeProps: {closeBrackets: {triples: '"'}} |
||||
}); |
||||
|
||||
def(["x-shader/x-vertex", "x-shader/x-fragment"], { |
||||
name: "clike", |
||||
keywords: words("sampler1D sampler2D sampler3D samplerCube " + |
||||
"sampler1DShadow sampler2DShadow " + |
||||
"const attribute uniform varying " + |
||||
"break continue discard return " + |
||||
"for while do if else struct " + |
||||
"in out inout"), |
||||
types: words("float int bool void " + |
||||
"vec2 vec3 vec4 ivec2 ivec3 ivec4 bvec2 bvec3 bvec4 " + |
||||
"mat2 mat3 mat4"), |
||||
blockKeywords: words("for while do if else struct"), |
||||
builtin: words("radians degrees sin cos tan asin acos atan " + |
||||
"pow exp log exp2 sqrt inversesqrt " + |
||||
"abs sign floor ceil fract mod min max clamp mix step smoothstep " + |
||||
"length distance dot cross normalize ftransform faceforward " + |
||||
"reflect refract matrixCompMult " + |
||||
"lessThan lessThanEqual greaterThan greaterThanEqual " + |
||||
"equal notEqual any all not " + |
||||
"texture1D texture1DProj texture1DLod texture1DProjLod " + |
||||
"texture2D texture2DProj texture2DLod texture2DProjLod " + |
||||
"texture3D texture3DProj texture3DLod texture3DProjLod " + |
||||
"textureCube textureCubeLod " + |
||||
"shadow1D shadow2D shadow1DProj shadow2DProj " + |
||||
"shadow1DLod shadow2DLod shadow1DProjLod shadow2DProjLod " + |
||||
"dFdx dFdy fwidth " + |
||||
"noise1 noise2 noise3 noise4"), |
||||
atoms: words("true false " + |
||||
"gl_FragColor gl_SecondaryColor gl_Normal gl_Vertex " + |
||||
"gl_MultiTexCoord0 gl_MultiTexCoord1 gl_MultiTexCoord2 gl_MultiTexCoord3 " + |
||||
"gl_MultiTexCoord4 gl_MultiTexCoord5 gl_MultiTexCoord6 gl_MultiTexCoord7 " + |
||||
"gl_FogCoord gl_PointCoord " + |
||||
"gl_Position gl_PointSize gl_ClipVertex " + |
||||
"gl_FrontColor gl_BackColor gl_FrontSecondaryColor gl_BackSecondaryColor " + |
||||
"gl_TexCoord gl_FogFragCoord " + |
||||
"gl_FragCoord gl_FrontFacing " + |
||||
"gl_FragData gl_FragDepth " + |
||||
"gl_ModelViewMatrix gl_ProjectionMatrix gl_ModelViewProjectionMatrix " + |
||||
"gl_TextureMatrix gl_NormalMatrix gl_ModelViewMatrixInverse " + |
||||
"gl_ProjectionMatrixInverse gl_ModelViewProjectionMatrixInverse " + |
||||
"gl_TexureMatrixTranspose gl_ModelViewMatrixInverseTranspose " + |
||||
"gl_ProjectionMatrixInverseTranspose " + |
||||
"gl_ModelViewProjectionMatrixInverseTranspose " + |
||||
"gl_TextureMatrixInverseTranspose " + |
||||
"gl_NormalScale gl_DepthRange gl_ClipPlane " + |
||||
"gl_Point gl_FrontMaterial gl_BackMaterial gl_LightSource gl_LightModel " + |
||||
"gl_FrontLightModelProduct gl_BackLightModelProduct " + |
||||
"gl_TextureColor gl_EyePlaneS gl_EyePlaneT gl_EyePlaneR gl_EyePlaneQ " + |
||||
"gl_FogParameters " + |
||||
"gl_MaxLights gl_MaxClipPlanes gl_MaxTextureUnits gl_MaxTextureCoords " + |
||||
"gl_MaxVertexAttribs gl_MaxVertexUniformComponents gl_MaxVaryingFloats " + |
||||
"gl_MaxVertexTextureImageUnits gl_MaxTextureImageUnits " + |
||||
"gl_MaxFragmentUniformComponents gl_MaxCombineTextureImageUnits " + |
||||
"gl_MaxDrawBuffers"), |
||||
indentSwitch: false, |
||||
hooks: {"#": cppHook}, |
||||
modeProps: {fold: ["brace", "include"]} |
||||
}); |
||||
|
||||
def("text/x-nesc", { |
||||
name: "clike", |
||||
keywords: words(cKeywords + "as atomic async call command component components configuration event generic " + |
||||
"implementation includes interface module new norace nx_struct nx_union post provides " + |
||||
"signal task uses abstract extends"), |
||||
types: words(cTypes), |
||||
blockKeywords: words("case do else for if switch while struct"), |
||||
atoms: words("null true false"), |
||||
hooks: {"#": cppHook}, |
||||
modeProps: {fold: ["brace", "include"]} |
||||
}); |
||||
|
||||
def("text/x-objectivec", { |
||||
name: "clike", |
||||
keywords: words(cKeywords + "inline restrict _Bool _Complex _Imaginary BOOL Class bycopy byref id IMP in " + |
||||
"inout nil oneway out Protocol SEL self super atomic nonatomic retain copy readwrite readonly"), |
||||
types: words(cTypes), |
||||
atoms: words("YES NO NULL NILL ON OFF true false"), |
||||
hooks: { |
||||
"@": function(stream) { |
||||
stream.eatWhile(/[\w\$]/); |
||||
return "keyword"; |
||||
}, |
||||
"#": cppHook, |
||||
indent: function(_state, ctx, textAfter) { |
||||
if (ctx.type == "statement" && /^@\w/.test(textAfter)) return ctx.indented |
||||
} |
||||
}, |
||||
modeProps: {fold: "brace"} |
||||
}); |
||||
|
||||
def("text/x-squirrel", { |
||||
name: "clike", |
||||
keywords: words("base break clone continue const default delete enum extends function in class" + |
||||
" foreach local resume return this throw typeof yield constructor instanceof static"), |
||||
types: words(cTypes), |
||||
blockKeywords: words("case catch class else for foreach if switch try while"), |
||||
defKeywords: words("function local class"), |
||||
typeFirstDefinitions: true, |
||||
atoms: words("true false null"), |
||||
hooks: {"#": cppHook}, |
||||
modeProps: {fold: ["brace", "include"]} |
||||
}); |
||||
|
||||
// Ceylon Strings need to deal with interpolation
|
||||
var stringTokenizer = null; |
||||
function tokenCeylonString(type) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while (!stream.eol()) { |
||||
if (!escaped && stream.match('"') && |
||||
(type == "single" || stream.match('""'))) { |
||||
end = true; |
||||
break; |
||||
} |
||||
if (!escaped && stream.match('``')) { |
||||
stringTokenizer = tokenCeylonString(type); |
||||
end = true; |
||||
break; |
||||
} |
||||
next = stream.next(); |
||||
escaped = type == "single" && !escaped && next == "\\"; |
||||
} |
||||
if (end) |
||||
state.tokenize = null; |
||||
return "string"; |
||||
} |
||||
} |
||||
|
||||
def("text/x-ceylon", { |
||||
name: "clike", |
||||
keywords: words("abstracts alias assembly assert assign break case catch class continue dynamic else" + |
||||
" exists extends finally for function given if import in interface is let module new" + |
||||
" nonempty object of out outer package return satisfies super switch then this throw" + |
||||
" try value void while"), |
||||
types: function(word) { |
||||
// In Ceylon all identifiers that start with an uppercase are types
|
||||
var first = word.charAt(0); |
||||
return (first === first.toUpperCase() && first !== first.toLowerCase()); |
||||
}, |
||||
blockKeywords: words("case catch class dynamic else finally for function if interface module new object switch try while"), |
||||
defKeywords: words("class dynamic function interface module object package value"), |
||||
builtin: words("abstract actual aliased annotation by default deprecated doc final formal late license" + |
||||
" native optional sealed see serializable shared suppressWarnings tagged throws variable"), |
||||
isPunctuationChar: /[\[\]{}\(\),;\:\.`]/, |
||||
isOperatorChar: /[+\-*&%=<>!?|^~:\/]/, |
||||
numberStart: /[\d#$]/, |
||||
number: /^(?:#[\da-fA-F_]+|\$[01_]+|[\d_]+[kMGTPmunpf]?|[\d_]+\.[\d_]+(?:[eE][-+]?\d+|[kMGTPmunpf]|)|)/i, |
||||
multiLineStrings: true, |
||||
typeFirstDefinitions: true, |
||||
atoms: words("true false null larger smaller equal empty finished"), |
||||
indentSwitch: false, |
||||
styleDefs: false, |
||||
hooks: { |
||||
"@": function(stream) { |
||||
stream.eatWhile(/[\w\$_]/); |
||||
return "meta"; |
||||
}, |
||||
'"': function(stream, state) { |
||||
state.tokenize = tokenCeylonString(stream.match('""') ? "triple" : "single"); |
||||
return state.tokenize(stream, state); |
||||
}, |
||||
'`': function(stream, state) { |
||||
if (!stringTokenizer || !stream.match('`')) return false; |
||||
state.tokenize = stringTokenizer; |
||||
stringTokenizer = null; |
||||
return state.tokenize(stream, state); |
||||
}, |
||||
"'": function(stream) { |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
return "atom"; |
||||
}, |
||||
token: function(_stream, state, style) { |
||||
if ((style == "variable" || style == "variable-3") && |
||||
state.prevToken == ".") { |
||||
return "variable-2"; |
||||
} |
||||
} |
||||
}, |
||||
modeProps: { |
||||
fold: ["brace", "import"], |
||||
closeBrackets: {triples: '"'} |
||||
} |
||||
}); |
||||
|
||||
}); |
@ -0,0 +1,360 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: C-like mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<link rel="stylesheet" href="../../addon/hint/show-hint.css"> |
||||
<script src="../../addon/hint/show-hint.js"></script> |
||||
<script src="clike.js"></script> |
||||
<style>.CodeMirror {border: 2px inset #dee;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">C-like</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>C-like mode</h2> |
||||
|
||||
<div><textarea id="c-code"> |
||||
/* C demo code */ |
||||
|
||||
#include <zmq.h> |
||||
#include <pthread.h> |
||||
#include <semaphore.h> |
||||
#include <time.h> |
||||
#include <stdio.h> |
||||
#include <fcntl.h> |
||||
#include <malloc.h> |
||||
|
||||
typedef struct { |
||||
void* arg_socket; |
||||
zmq_msg_t* arg_msg; |
||||
char* arg_string; |
||||
unsigned long arg_len; |
||||
int arg_int, arg_command; |
||||
|
||||
int signal_fd; |
||||
int pad; |
||||
void* context; |
||||
sem_t sem; |
||||
} acl_zmq_context; |
||||
|
||||
#define p(X) (context->arg_##X) |
||||
|
||||
void* zmq_thread(void* context_pointer) { |
||||
acl_zmq_context* context = (acl_zmq_context*)context_pointer; |
||||
char ok = 'K', err = 'X'; |
||||
int res; |
||||
|
||||
while (1) { |
||||
while ((res = sem_wait(&context->sem)) == EINTR); |
||||
if (res) {write(context->signal_fd, &err, 1); goto cleanup;} |
||||
switch(p(command)) { |
||||
case 0: goto cleanup; |
||||
case 1: p(socket) = zmq_socket(context->context, p(int)); break; |
||||
case 2: p(int) = zmq_close(p(socket)); break; |
||||
case 3: p(int) = zmq_bind(p(socket), p(string)); break; |
||||
case 4: p(int) = zmq_connect(p(socket), p(string)); break; |
||||
case 5: p(int) = zmq_getsockopt(p(socket), p(int), (void*)p(string), &p(len)); break; |
||||
case 6: p(int) = zmq_setsockopt(p(socket), p(int), (void*)p(string), p(len)); break; |
||||
case 7: p(int) = zmq_send(p(socket), p(msg), p(int)); break; |
||||
case 8: p(int) = zmq_recv(p(socket), p(msg), p(int)); break; |
||||
case 9: p(int) = zmq_poll(p(socket), p(int), p(len)); break; |
||||
} |
||||
p(command) = errno; |
||||
write(context->signal_fd, &ok, 1); |
||||
} |
||||
cleanup: |
||||
close(context->signal_fd); |
||||
free(context_pointer); |
||||
return 0; |
||||
} |
||||
|
||||
void* zmq_thread_init(void* zmq_context, int signal_fd) { |
||||
acl_zmq_context* context = malloc(sizeof(acl_zmq_context)); |
||||
pthread_t thread; |
||||
|
||||
context->context = zmq_context; |
||||
context->signal_fd = signal_fd; |
||||
sem_init(&context->sem, 1, 0); |
||||
pthread_create(&thread, 0, &zmq_thread, context); |
||||
pthread_detach(thread); |
||||
return context; |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>C++ example</h2> |
||||
|
||||
<div><textarea id="cpp-code"> |
||||
#include <iostream> |
||||
#include "mystuff/util.h" |
||||
|
||||
namespace { |
||||
enum Enum { |
||||
VAL1, VAL2, VAL3 |
||||
}; |
||||
|
||||
char32_t unicode_string = U"\U0010FFFF"; |
||||
string raw_string = R"delim(anything |
||||
you |
||||
want)delim"; |
||||
|
||||
int Helper(const MyType& param) { |
||||
return 0; |
||||
} |
||||
} // namespace |
||||
|
||||
class ForwardDec; |
||||
|
||||
template <class T, class V> |
||||
class Class : public BaseClass { |
||||
const MyType<T, V> member_; |
||||
|
||||
public: |
||||
const MyType<T, V>& Method() const { |
||||
return member_; |
||||
} |
||||
|
||||
void Method2(MyType<T, V>* value); |
||||
} |
||||
|
||||
template <class T, class V> |
||||
void Class::Method2(MyType<T, V>* value) { |
||||
std::out << 1 >> method(); |
||||
value->Method3(member_); |
||||
member_ = value; |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>Objective-C example</h2> |
||||
|
||||
<div><textarea id="objectivec-code"> |
||||
/* |
||||
This is a longer comment |
||||
That spans two lines |
||||
*/ |
||||
|
||||
#import <Test/Test.h> |
||||
@implementation YourAppDelegate |
||||
|
||||
// This is a one-line comment |
||||
|
||||
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{ |
||||
char myString[] = "This is a C character array"; |
||||
int test = 5; |
||||
return YES; |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>Java example</h2> |
||||
|
||||
<div><textarea id="java-code"> |
||||
import com.demo.util.MyType; |
||||
import com.demo.util.MyInterface; |
||||
|
||||
public enum Enum { |
||||
VAL1, VAL2, VAL3 |
||||
} |
||||
|
||||
public class Class<T, V> implements MyInterface { |
||||
public static final MyType<T, V> member; |
||||
|
||||
private class InnerClass { |
||||
public int zero() { |
||||
return 0; |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public MyType method() { |
||||
return member; |
||||
} |
||||
|
||||
public void method2(MyType<T, V> value) { |
||||
method(); |
||||
value.method3(); |
||||
member = value; |
||||
} |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>Scala example</h2> |
||||
|
||||
<div><textarea id="scala-code"> |
||||
object FilterTest extends App { |
||||
def filter(xs: List[Int], threshold: Int) = { |
||||
def process(ys: List[Int]): List[Int] = |
||||
if (ys.isEmpty) ys |
||||
else if (ys.head < threshold) ys.head :: process(ys.tail) |
||||
else process(ys.tail) |
||||
process(xs) |
||||
} |
||||
println(filter(List(1, 9, 2, 8, 3, 7, 4), 5)) |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>Kotlin mode</h2> |
||||
|
||||
<div><textarea id="kotlin-code"> |
||||
package org.wasabi.http |
||||
|
||||
import java.util.concurrent.Executors |
||||
import java.net.InetSocketAddress |
||||
import org.wasabi.app.AppConfiguration |
||||
import io.netty.bootstrap.ServerBootstrap |
||||
import io.netty.channel.nio.NioEventLoopGroup |
||||
import io.netty.channel.socket.nio.NioServerSocketChannel |
||||
import org.wasabi.app.AppServer |
||||
|
||||
public class HttpServer(private val appServer: AppServer) { |
||||
|
||||
val bootstrap: ServerBootstrap |
||||
val primaryGroup: NioEventLoopGroup |
||||
val workerGroup: NioEventLoopGroup |
||||
|
||||
init { |
||||
// Define worker groups |
||||
primaryGroup = NioEventLoopGroup() |
||||
workerGroup = NioEventLoopGroup() |
||||
|
||||
// Initialize bootstrap of server |
||||
bootstrap = ServerBootstrap() |
||||
|
||||
bootstrap.group(primaryGroup, workerGroup) |
||||
bootstrap.channel(javaClass<NioServerSocketChannel>()) |
||||
bootstrap.childHandler(NettyPipelineInitializer(appServer)) |
||||
} |
||||
|
||||
public fun start(wait: Boolean = true) { |
||||
val channel = bootstrap.bind(appServer.configuration.port)?.sync()?.channel() |
||||
|
||||
if (wait) { |
||||
channel?.closeFuture()?.sync() |
||||
} |
||||
} |
||||
|
||||
public fun stop() { |
||||
// Shutdown all event loops |
||||
primaryGroup.shutdownGracefully() |
||||
workerGroup.shutdownGracefully() |
||||
|
||||
// Wait till all threads are terminated |
||||
primaryGroup.terminationFuture().sync() |
||||
workerGroup.terminationFuture().sync() |
||||
} |
||||
} |
||||
</textarea></div> |
||||
|
||||
<h2>Ceylon mode</h2> |
||||
|
||||
<div><textarea id="ceylon-code"> |
||||
"Produces the [[stream|Iterable]] that results from repeated |
||||
application of the given [[function|next]] to the given |
||||
[[first]] element of the stream, until the function first |
||||
returns [[finished]]. If the given function never returns |
||||
`finished`, the resulting stream is infinite. |
||||
|
||||
For example: |
||||
|
||||
loop(0)(2.plus).takeWhile(10.largerThan) |
||||
|
||||
produces the stream `{ 0, 2, 4, 6, 8 }`." |
||||
tagged("Streams") |
||||
shared {Element+} loop<Element>( |
||||
"The first element of the resulting stream." |
||||
Element first)( |
||||
"The function that produces the next element of the |
||||
stream, given the current element. The function may |
||||
return [[finished]] to indicate the end of the |
||||
stream." |
||||
Element|Finished next(Element element)) |
||||
=> let (start = first) |
||||
object satisfies {Element+} { |
||||
first => start; |
||||
empty => false; |
||||
function nextElement(Element element) |
||||
=> next(element); |
||||
iterator() |
||||
=> object satisfies Iterator<Element> { |
||||
variable Element|Finished current = start; |
||||
shared actual Element|Finished next() { |
||||
if (!is Finished result = current) { |
||||
current = nextElement(result); |
||||
return result; |
||||
} |
||||
else { |
||||
return finished; |
||||
} |
||||
} |
||||
}; |
||||
}; |
||||
</textarea></div> |
||||
|
||||
<script> |
||||
var cEditor = CodeMirror.fromTextArea(document.getElementById("c-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-csrc" |
||||
}); |
||||
var cppEditor = CodeMirror.fromTextArea(document.getElementById("cpp-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-c++src" |
||||
}); |
||||
var javaEditor = CodeMirror.fromTextArea(document.getElementById("java-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-java" |
||||
}); |
||||
var objectivecEditor = CodeMirror.fromTextArea(document.getElementById("objectivec-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-objectivec" |
||||
}); |
||||
var scalaEditor = CodeMirror.fromTextArea(document.getElementById("scala-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-scala" |
||||
}); |
||||
var kotlinEditor = CodeMirror.fromTextArea(document.getElementById("kotlin-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-kotlin" |
||||
}); |
||||
var ceylonEditor = CodeMirror.fromTextArea(document.getElementById("ceylon-code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-ceylon" |
||||
}); |
||||
var mac = CodeMirror.keyMap.default == CodeMirror.keyMap.macDefault; |
||||
CodeMirror.keyMap.default[(mac ? "Cmd" : "Ctrl") + "-Space"] = "autocomplete"; |
||||
</script> |
||||
|
||||
<p>Simple mode that tries to handle C-like languages as well as it |
||||
can. Takes two configuration parameters: <code>keywords</code>, an |
||||
object whose property names are the keywords in the language, |
||||
and <code>useCPP</code>, which determines whether C preprocessor |
||||
directives are recognized.</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-csrc</code> |
||||
(C), <code>text/x-c++src</code> (C++), <code>text/x-java</code> |
||||
(Java), <code>text/x-csharp</code> (C#), |
||||
<code>text/x-objectivec</code> (Objective-C), |
||||
<code>text/x-scala</code> (Scala), <code>text/x-vertex</code> |
||||
<code>x-shader/x-fragment</code> (shader programs), |
||||
<code>text/x-squirrel</code> (Squirrel) and |
||||
<code>text/x-ceylon</code> (Ceylon)</p> |
||||
</article> |
@ -0,0 +1,767 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Scala mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/ambiance.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="clike.js"></script> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Scala</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Scala mode</h2> |
||||
<form> |
||||
<textarea id="code" name="code"> |
||||
|
||||
/* __ *\ |
||||
** ________ ___ / / ___ Scala API ** |
||||
** / __/ __// _ | / / / _ | (c) 2003-2011, LAMP/EPFL ** |
||||
** __\ \/ /__/ __ |/ /__/ __ | http://scala-lang.org/ ** |
||||
** /____/\___/_/ |_/____/_/ | | ** |
||||
** |/ ** |
||||
\* */ |
||||
|
||||
package scala.collection |
||||
|
||||
import generic._ |
||||
import mutable.{ Builder, ListBuffer } |
||||
import annotation.{tailrec, migration, bridge} |
||||
import annotation.unchecked.{ uncheckedVariance => uV } |
||||
import parallel.ParIterable |
||||
|
||||
/** A template trait for traversable collections of type `Traversable[A]`. |
||||
* |
||||
* $traversableInfo |
||||
* @define mutability |
||||
* @define traversableInfo |
||||
* This is a base trait of all kinds of $mutability Scala collections. It |
||||
* implements the behavior common to all collections, in terms of a method |
||||
* `foreach` with signature: |
||||
* {{{ |
||||
* def foreach[U](f: Elem => U): Unit |
||||
* }}} |
||||
* Collection classes mixing in this trait provide a concrete |
||||
* `foreach` method which traverses all the |
||||
* elements contained in the collection, applying a given function to each. |
||||
* They also need to provide a method `newBuilder` |
||||
* which creates a builder for collections of the same kind. |
||||
* |
||||
* A traversable class might or might not have two properties: strictness |
||||
* and orderedness. Neither is represented as a type. |
||||
* |
||||
* The instances of a strict collection class have all their elements |
||||
* computed before they can be used as values. By contrast, instances of |
||||
* a non-strict collection class may defer computation of some of their |
||||
* elements until after the instance is available as a value. |
||||
* A typical example of a non-strict collection class is a |
||||
* <a href="../immutable/Stream.html" target="ContentFrame"> |
||||
* `scala.collection.immutable.Stream`</a>. |
||||
* A more general class of examples are `TraversableViews`. |
||||
* |
||||
* If a collection is an instance of an ordered collection class, traversing |
||||
* its elements with `foreach` will always visit elements in the |
||||
* same order, even for different runs of the program. If the class is not |
||||
* ordered, `foreach` can visit elements in different orders for |
||||
* different runs (but it will keep the same order in the same run).' |
||||
* |
||||
* A typical example of a collection class which is not ordered is a |
||||
* `HashMap` of objects. The traversal order for hash maps will |
||||
* depend on the hash codes of its elements, and these hash codes might |
||||
* differ from one run to the next. By contrast, a `LinkedHashMap` |
||||
* is ordered because it's `foreach` method visits elements in the |
||||
* order they were inserted into the `HashMap`. |
||||
* |
||||
* @author Martin Odersky |
||||
* @version 2.8 |
||||
* @since 2.8 |
||||
* @tparam A the element type of the collection |
||||
* @tparam Repr the type of the actual collection containing the elements. |
||||
* |
||||
* @define Coll Traversable |
||||
* @define coll traversable collection |
||||
*/ |
||||
trait TraversableLike[+A, +Repr] extends HasNewBuilder[A, Repr] |
||||
with FilterMonadic[A, Repr] |
||||
with TraversableOnce[A] |
||||
with GenTraversableLike[A, Repr] |
||||
with Parallelizable[A, ParIterable[A]] |
||||
{ |
||||
self => |
||||
|
||||
import Traversable.breaks._ |
||||
|
||||
/** The type implementing this traversable */ |
||||
protected type Self = Repr |
||||
|
||||
/** The collection of type $coll underlying this `TraversableLike` object. |
||||
* By default this is implemented as the `TraversableLike` object itself, |
||||
* but this can be overridden. |
||||
*/ |
||||
def repr: Repr = this.asInstanceOf[Repr] |
||||
|
||||
/** The underlying collection seen as an instance of `$Coll`. |
||||
* By default this is implemented as the current collection object itself, |
||||
* but this can be overridden. |
||||
*/ |
||||
protected[this] def thisCollection: Traversable[A] = this.asInstanceOf[Traversable[A]] |
||||
|
||||
/** A conversion from collections of type `Repr` to `$Coll` objects. |
||||
* By default this is implemented as just a cast, but this can be overridden. |
||||
*/ |
||||
protected[this] def toCollection(repr: Repr): Traversable[A] = repr.asInstanceOf[Traversable[A]] |
||||
|
||||
/** Creates a new builder for this collection type. |
||||
*/ |
||||
protected[this] def newBuilder: Builder[A, Repr] |
||||
|
||||
protected[this] def parCombiner = ParIterable.newCombiner[A] |
||||
|
||||
/** Applies a function `f` to all elements of this $coll. |
||||
* |
||||
* Note: this method underlies the implementation of most other bulk operations. |
||||
* It's important to implement this method in an efficient way. |
||||
* |
||||
* |
||||
* @param f the function that is applied for its side-effect to every element. |
||||
* The result of function `f` is discarded. |
||||
* |
||||
* @tparam U the type parameter describing the result of function `f`. |
||||
* This result will always be ignored. Typically `U` is `Unit`, |
||||
* but this is not necessary. |
||||
* |
||||
* @usecase def foreach(f: A => Unit): Unit |
||||
*/ |
||||
def foreach[U](f: A => U): Unit |
||||
|
||||
/** Tests whether this $coll is empty. |
||||
* |
||||
* @return `true` if the $coll contain no elements, `false` otherwise. |
||||
*/ |
||||
def isEmpty: Boolean = { |
||||
var result = true |
||||
breakable { |
||||
for (x <- this) { |
||||
result = false |
||||
break |
||||
} |
||||
} |
||||
result |
||||
} |
||||
|
||||
/** Tests whether this $coll is known to have a finite size. |
||||
* All strict collections are known to have finite size. For a non-strict collection |
||||
* such as `Stream`, the predicate returns `true` if all elements have been computed. |
||||
* It returns `false` if the stream is not yet evaluated to the end. |
||||
* |
||||
* Note: many collection methods will not work on collections of infinite sizes. |
||||
* |
||||
* @return `true` if this collection is known to have finite size, `false` otherwise. |
||||
*/ |
||||
def hasDefiniteSize = true |
||||
|
||||
def ++[B >: A, That](that: GenTraversableOnce[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
if (that.isInstanceOf[IndexedSeqLike[_, _]]) b.sizeHint(this, that.seq.size) |
||||
b ++= thisCollection |
||||
b ++= that.seq |
||||
b.result |
||||
} |
||||
|
||||
@bridge |
||||
def ++[B >: A, That](that: TraversableOnce[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = |
||||
++(that: GenTraversableOnce[B])(bf) |
||||
|
||||
/** Concatenates this $coll with the elements of a traversable collection. |
||||
* It differs from ++ in that the right operand determines the type of the |
||||
* resulting collection rather than the left one. |
||||
* |
||||
* @param that the traversable to append. |
||||
* @tparam B the element type of the returned collection. |
||||
* @tparam That $thatinfo |
||||
* @param bf $bfinfo |
||||
* @return a new collection of type `That` which contains all elements |
||||
* of this $coll followed by all elements of `that`. |
||||
* |
||||
* @usecase def ++:[B](that: TraversableOnce[B]): $Coll[B] |
||||
* |
||||
* @return a new $coll which contains all elements of this $coll |
||||
* followed by all elements of `that`. |
||||
*/ |
||||
def ++:[B >: A, That](that: TraversableOnce[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
if (that.isInstanceOf[IndexedSeqLike[_, _]]) b.sizeHint(this, that.size) |
||||
b ++= that |
||||
b ++= thisCollection |
||||
b.result |
||||
} |
||||
|
||||
/** This overload exists because: for the implementation of ++: we should reuse |
||||
* that of ++ because many collections override it with more efficient versions. |
||||
* Since TraversableOnce has no '++' method, we have to implement that directly, |
||||
* but Traversable and down can use the overload. |
||||
*/ |
||||
def ++:[B >: A, That](that: Traversable[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = |
||||
(that ++ seq)(breakOut) |
||||
|
||||
def map[B, That](f: A => B)(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
b.sizeHint(this) |
||||
for (x <- this) b += f(x) |
||||
b.result |
||||
} |
||||
|
||||
def flatMap[B, That](f: A => GenTraversableOnce[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
for (x <- this) b ++= f(x).seq |
||||
b.result |
||||
} |
||||
|
||||
/** Selects all elements of this $coll which satisfy a predicate. |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return a new $coll consisting of all elements of this $coll that satisfy the given |
||||
* predicate `p`. The order of the elements is preserved. |
||||
*/ |
||||
def filter(p: A => Boolean): Repr = { |
||||
val b = newBuilder |
||||
for (x <- this) |
||||
if (p(x)) b += x |
||||
b.result |
||||
} |
||||
|
||||
/** Selects all elements of this $coll which do not satisfy a predicate. |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return a new $coll consisting of all elements of this $coll that do not satisfy the given |
||||
* predicate `p`. The order of the elements is preserved. |
||||
*/ |
||||
def filterNot(p: A => Boolean): Repr = filter(!p(_)) |
||||
|
||||
def collect[B, That](pf: PartialFunction[A, B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
for (x <- this) if (pf.isDefinedAt(x)) b += pf(x) |
||||
b.result |
||||
} |
||||
|
||||
/** Builds a new collection by applying an option-valued function to all |
||||
* elements of this $coll on which the function is defined. |
||||
* |
||||
* @param f the option-valued function which filters and maps the $coll. |
||||
* @tparam B the element type of the returned collection. |
||||
* @tparam That $thatinfo |
||||
* @param bf $bfinfo |
||||
* @return a new collection of type `That` resulting from applying the option-valued function |
||||
* `f` to each element and collecting all defined results. |
||||
* The order of the elements is preserved. |
||||
* |
||||
* @usecase def filterMap[B](f: A => Option[B]): $Coll[B] |
||||
* |
||||
* @param pf the partial function which filters and maps the $coll. |
||||
* @return a new $coll resulting from applying the given option-valued function |
||||
* `f` to each element and collecting all defined results. |
||||
* The order of the elements is preserved. |
||||
def filterMap[B, That](f: A => Option[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
for (x <- this) |
||||
f(x) match { |
||||
case Some(y) => b += y |
||||
case _ => |
||||
} |
||||
b.result |
||||
} |
||||
*/ |
||||
|
||||
/** Partitions this $coll in two ${coll}s according to a predicate. |
||||
* |
||||
* @param p the predicate on which to partition. |
||||
* @return a pair of ${coll}s: the first $coll consists of all elements that |
||||
* satisfy the predicate `p` and the second $coll consists of all elements |
||||
* that don't. The relative order of the elements in the resulting ${coll}s |
||||
* is the same as in the original $coll. |
||||
*/ |
||||
def partition(p: A => Boolean): (Repr, Repr) = { |
||||
val l, r = newBuilder |
||||
for (x <- this) (if (p(x)) l else r) += x |
||||
(l.result, r.result) |
||||
} |
||||
|
||||
def groupBy[K](f: A => K): immutable.Map[K, Repr] = { |
||||
val m = mutable.Map.empty[K, Builder[A, Repr]] |
||||
for (elem <- this) { |
||||
val key = f(elem) |
||||
val bldr = m.getOrElseUpdate(key, newBuilder) |
||||
bldr += elem |
||||
} |
||||
val b = immutable.Map.newBuilder[K, Repr] |
||||
for ((k, v) <- m) |
||||
b += ((k, v.result)) |
||||
|
||||
b.result |
||||
} |
||||
|
||||
/** Tests whether a predicate holds for all elements of this $coll. |
||||
* |
||||
* $mayNotTerminateInf |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return `true` if the given predicate `p` holds for all elements |
||||
* of this $coll, otherwise `false`. |
||||
*/ |
||||
def forall(p: A => Boolean): Boolean = { |
||||
var result = true |
||||
breakable { |
||||
for (x <- this) |
||||
if (!p(x)) { result = false; break } |
||||
} |
||||
result |
||||
} |
||||
|
||||
/** Tests whether a predicate holds for some of the elements of this $coll. |
||||
* |
||||
* $mayNotTerminateInf |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return `true` if the given predicate `p` holds for some of the |
||||
* elements of this $coll, otherwise `false`. |
||||
*/ |
||||
def exists(p: A => Boolean): Boolean = { |
||||
var result = false |
||||
breakable { |
||||
for (x <- this) |
||||
if (p(x)) { result = true; break } |
||||
} |
||||
result |
||||
} |
||||
|
||||
/** Finds the first element of the $coll satisfying a predicate, if any. |
||||
* |
||||
* $mayNotTerminateInf |
||||
* $orderDependent |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return an option value containing the first element in the $coll |
||||
* that satisfies `p`, or `None` if none exists. |
||||
*/ |
||||
def find(p: A => Boolean): Option[A] = { |
||||
var result: Option[A] = None |
||||
breakable { |
||||
for (x <- this) |
||||
if (p(x)) { result = Some(x); break } |
||||
} |
||||
result |
||||
} |
||||
|
||||
def scan[B >: A, That](z: B)(op: (B, B) => B)(implicit cbf: CanBuildFrom[Repr, B, That]): That = scanLeft(z)(op) |
||||
|
||||
def scanLeft[B, That](z: B)(op: (B, A) => B)(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
b.sizeHint(this, 1) |
||||
var acc = z |
||||
b += acc |
||||
for (x <- this) { acc = op(acc, x); b += acc } |
||||
b.result |
||||
} |
||||
|
||||
@migration(2, 9, |
||||
"This scanRight definition has changed in 2.9.\n" + |
||||
"The previous behavior can be reproduced with scanRight.reverse." |
||||
) |
||||
def scanRight[B, That](z: B)(op: (A, B) => B)(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
var scanned = List(z) |
||||
var acc = z |
||||
for (x <- reversed) { |
||||
acc = op(x, acc) |
||||
scanned ::= acc |
||||
} |
||||
val b = bf(repr) |
||||
for (elem <- scanned) b += elem |
||||
b.result |
||||
} |
||||
|
||||
/** Selects the first element of this $coll. |
||||
* $orderDependent |
||||
* @return the first element of this $coll. |
||||
* @throws `NoSuchElementException` if the $coll is empty. |
||||
*/ |
||||
def head: A = { |
||||
var result: () => A = () => throw new NoSuchElementException |
||||
breakable { |
||||
for (x <- this) { |
||||
result = () => x |
||||
break |
||||
} |
||||
} |
||||
result() |
||||
} |
||||
|
||||
/** Optionally selects the first element. |
||||
* $orderDependent |
||||
* @return the first element of this $coll if it is nonempty, `None` if it is empty. |
||||
*/ |
||||
def headOption: Option[A] = if (isEmpty) None else Some(head) |
||||
|
||||
/** Selects all elements except the first. |
||||
* $orderDependent |
||||
* @return a $coll consisting of all elements of this $coll |
||||
* except the first one. |
||||
* @throws `UnsupportedOperationException` if the $coll is empty. |
||||
*/ |
||||
override def tail: Repr = { |
||||
if (isEmpty) throw new UnsupportedOperationException("empty.tail") |
||||
drop(1) |
||||
} |
||||
|
||||
/** Selects the last element. |
||||
* $orderDependent |
||||
* @return The last element of this $coll. |
||||
* @throws NoSuchElementException If the $coll is empty. |
||||
*/ |
||||
def last: A = { |
||||
var lst = head |
||||
for (x <- this) |
||||
lst = x |
||||
lst |
||||
} |
||||
|
||||
/** Optionally selects the last element. |
||||
* $orderDependent |
||||
* @return the last element of this $coll$ if it is nonempty, `None` if it is empty. |
||||
*/ |
||||
def lastOption: Option[A] = if (isEmpty) None else Some(last) |
||||
|
||||
/** Selects all elements except the last. |
||||
* $orderDependent |
||||
* @return a $coll consisting of all elements of this $coll |
||||
* except the last one. |
||||
* @throws `UnsupportedOperationException` if the $coll is empty. |
||||
*/ |
||||
def init: Repr = { |
||||
if (isEmpty) throw new UnsupportedOperationException("empty.init") |
||||
var lst = head |
||||
var follow = false |
||||
val b = newBuilder |
||||
b.sizeHint(this, -1) |
||||
for (x <- this.seq) { |
||||
if (follow) b += lst |
||||
else follow = true |
||||
lst = x |
||||
} |
||||
b.result |
||||
} |
||||
|
||||
def take(n: Int): Repr = slice(0, n) |
||||
|
||||
def drop(n: Int): Repr = |
||||
if (n <= 0) { |
||||
val b = newBuilder |
||||
b.sizeHint(this) |
||||
b ++= thisCollection result |
||||
} |
||||
else sliceWithKnownDelta(n, Int.MaxValue, -n) |
||||
|
||||
def slice(from: Int, until: Int): Repr = sliceWithKnownBound(math.max(from, 0), until) |
||||
|
||||
// Precondition: from >= 0, until > 0, builder already configured for building. |
||||
private[this] def sliceInternal(from: Int, until: Int, b: Builder[A, Repr]): Repr = { |
||||
var i = 0 |
||||
breakable { |
||||
for (x <- this.seq) { |
||||
if (i >= from) b += x |
||||
i += 1 |
||||
if (i >= until) break |
||||
} |
||||
} |
||||
b.result |
||||
} |
||||
// Precondition: from >= 0 |
||||
private[scala] def sliceWithKnownDelta(from: Int, until: Int, delta: Int): Repr = { |
||||
val b = newBuilder |
||||
if (until <= from) b.result |
||||
else { |
||||
b.sizeHint(this, delta) |
||||
sliceInternal(from, until, b) |
||||
} |
||||
} |
||||
// Precondition: from >= 0 |
||||
private[scala] def sliceWithKnownBound(from: Int, until: Int): Repr = { |
||||
val b = newBuilder |
||||
if (until <= from) b.result |
||||
else { |
||||
b.sizeHintBounded(until - from, this) |
||||
sliceInternal(from, until, b) |
||||
} |
||||
} |
||||
|
||||
def takeWhile(p: A => Boolean): Repr = { |
||||
val b = newBuilder |
||||
breakable { |
||||
for (x <- this) { |
||||
if (!p(x)) break |
||||
b += x |
||||
} |
||||
} |
||||
b.result |
||||
} |
||||
|
||||
def dropWhile(p: A => Boolean): Repr = { |
||||
val b = newBuilder |
||||
var go = false |
||||
for (x <- this) { |
||||
if (!p(x)) go = true |
||||
if (go) b += x |
||||
} |
||||
b.result |
||||
} |
||||
|
||||
def span(p: A => Boolean): (Repr, Repr) = { |
||||
val l, r = newBuilder |
||||
var toLeft = true |
||||
for (x <- this) { |
||||
toLeft = toLeft && p(x) |
||||
(if (toLeft) l else r) += x |
||||
} |
||||
(l.result, r.result) |
||||
} |
||||
|
||||
def splitAt(n: Int): (Repr, Repr) = { |
||||
val l, r = newBuilder |
||||
l.sizeHintBounded(n, this) |
||||
if (n >= 0) r.sizeHint(this, -n) |
||||
var i = 0 |
||||
for (x <- this) { |
||||
(if (i < n) l else r) += x |
||||
i += 1 |
||||
} |
||||
(l.result, r.result) |
||||
} |
||||
|
||||
/** Iterates over the tails of this $coll. The first value will be this |
||||
* $coll and the final one will be an empty $coll, with the intervening |
||||
* values the results of successive applications of `tail`. |
||||
* |
||||
* @return an iterator over all the tails of this $coll |
||||
* @example `List(1,2,3).tails = Iterator(List(1,2,3), List(2,3), List(3), Nil)` |
||||
*/ |
||||
def tails: Iterator[Repr] = iterateUntilEmpty(_.tail) |
||||
|
||||
/** Iterates over the inits of this $coll. The first value will be this |
||||
* $coll and the final one will be an empty $coll, with the intervening |
||||
* values the results of successive applications of `init`. |
||||
* |
||||
* @return an iterator over all the inits of this $coll |
||||
* @example `List(1,2,3).inits = Iterator(List(1,2,3), List(1,2), List(1), Nil)` |
||||
*/ |
||||
def inits: Iterator[Repr] = iterateUntilEmpty(_.init) |
||||
|
||||
/** Copies elements of this $coll to an array. |
||||
* Fills the given array `xs` with at most `len` elements of |
||||
* this $coll, starting at position `start`. |
||||
* Copying will stop once either the end of the current $coll is reached, |
||||
* or the end of the array is reached, or `len` elements have been copied. |
||||
* |
||||
* $willNotTerminateInf |
||||
* |
||||
* @param xs the array to fill. |
||||
* @param start the starting index. |
||||
* @param len the maximal number of elements to copy. |
||||
* @tparam B the type of the elements of the array. |
||||
* |
||||
* |
||||
* @usecase def copyToArray(xs: Array[A], start: Int, len: Int): Unit |
||||
*/ |
||||
def copyToArray[B >: A](xs: Array[B], start: Int, len: Int) { |
||||
var i = start |
||||
val end = (start + len) min xs.length |
||||
breakable { |
||||
for (x <- this) { |
||||
if (i >= end) break |
||||
xs(i) = x |
||||
i += 1 |
||||
} |
||||
} |
||||
} |
||||
|
||||
def toTraversable: Traversable[A] = thisCollection |
||||
def toIterator: Iterator[A] = toStream.iterator |
||||
def toStream: Stream[A] = toBuffer.toStream |
||||
|
||||
/** Converts this $coll to a string. |
||||
* |
||||
* @return a string representation of this collection. By default this |
||||
* string consists of the `stringPrefix` of this $coll, |
||||
* followed by all elements separated by commas and enclosed in parentheses. |
||||
*/ |
||||
override def toString = mkString(stringPrefix + "(", ", ", ")") |
||||
|
||||
/** Defines the prefix of this object's `toString` representation. |
||||
* |
||||
* @return a string representation which starts the result of `toString` |
||||
* applied to this $coll. By default the string prefix is the |
||||
* simple name of the collection class $coll. |
||||
*/ |
||||
def stringPrefix : String = { |
||||
var string = repr.asInstanceOf[AnyRef].getClass.getName |
||||
val idx1 = string.lastIndexOf('.' : Int) |
||||
if (idx1 != -1) string = string.substring(idx1 + 1) |
||||
val idx2 = string.indexOf('$') |
||||
if (idx2 != -1) string = string.substring(0, idx2) |
||||
string |
||||
} |
||||
|
||||
/** Creates a non-strict view of this $coll. |
||||
* |
||||
* @return a non-strict view of this $coll. |
||||
*/ |
||||
def view = new TraversableView[A, Repr] { |
||||
protected lazy val underlying = self.repr |
||||
override def foreach[U](f: A => U) = self foreach f |
||||
} |
||||
|
||||
/** Creates a non-strict view of a slice of this $coll. |
||||
* |
||||
* Note: the difference between `view` and `slice` is that `view` produces |
||||
* a view of the current $coll, whereas `slice` produces a new $coll. |
||||
* |
||||
* Note: `view(from, to)` is equivalent to `view.slice(from, to)` |
||||
* $orderDependent |
||||
* |
||||
* @param from the index of the first element of the view |
||||
* @param until the index of the element following the view |
||||
* @return a non-strict view of a slice of this $coll, starting at index `from` |
||||
* and extending up to (but not including) index `until`. |
||||
*/ |
||||
def view(from: Int, until: Int): TraversableView[A, Repr] = view.slice(from, until) |
||||
|
||||
/** Creates a non-strict filter of this $coll. |
||||
* |
||||
* Note: the difference between `c filter p` and `c withFilter p` is that |
||||
* the former creates a new collection, whereas the latter only |
||||
* restricts the domain of subsequent `map`, `flatMap`, `foreach`, |
||||
* and `withFilter` operations. |
||||
* $orderDependent |
||||
* |
||||
* @param p the predicate used to test elements. |
||||
* @return an object of class `WithFilter`, which supports |
||||
* `map`, `flatMap`, `foreach`, and `withFilter` operations. |
||||
* All these operations apply to those elements of this $coll which |
||||
* satisfy the predicate `p`. |
||||
*/ |
||||
def withFilter(p: A => Boolean): FilterMonadic[A, Repr] = new WithFilter(p) |
||||
|
||||
/** A class supporting filtered operations. Instances of this class are |
||||
* returned by method `withFilter`. |
||||
*/ |
||||
class WithFilter(p: A => Boolean) extends FilterMonadic[A, Repr] { |
||||
|
||||
/** Builds a new collection by applying a function to all elements of the |
||||
* outer $coll containing this `WithFilter` instance that satisfy predicate `p`. |
||||
* |
||||
* @param f the function to apply to each element. |
||||
* @tparam B the element type of the returned collection. |
||||
* @tparam That $thatinfo |
||||
* @param bf $bfinfo |
||||
* @return a new collection of type `That` resulting from applying |
||||
* the given function `f` to each element of the outer $coll |
||||
* that satisfies predicate `p` and collecting the results. |
||||
* |
||||
* @usecase def map[B](f: A => B): $Coll[B] |
||||
* |
||||
* @return a new $coll resulting from applying the given function |
||||
* `f` to each element of the outer $coll that satisfies |
||||
* predicate `p` and collecting the results. |
||||
*/ |
||||
def map[B, That](f: A => B)(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
for (x <- self) |
||||
if (p(x)) b += f(x) |
||||
b.result |
||||
} |
||||
|
||||
/** Builds a new collection by applying a function to all elements of the |
||||
* outer $coll containing this `WithFilter` instance that satisfy |
||||
* predicate `p` and concatenating the results. |
||||
* |
||||
* @param f the function to apply to each element. |
||||
* @tparam B the element type of the returned collection. |
||||
* @tparam That $thatinfo |
||||
* @param bf $bfinfo |
||||
* @return a new collection of type `That` resulting from applying |
||||
* the given collection-valued function `f` to each element |
||||
* of the outer $coll that satisfies predicate `p` and |
||||
* concatenating the results. |
||||
* |
||||
* @usecase def flatMap[B](f: A => TraversableOnce[B]): $Coll[B] |
||||
* |
||||
* @return a new $coll resulting from applying the given collection-valued function |
||||
* `f` to each element of the outer $coll that satisfies predicate `p` and concatenating the results. |
||||
*/ |
||||
def flatMap[B, That](f: A => GenTraversableOnce[B])(implicit bf: CanBuildFrom[Repr, B, That]): That = { |
||||
val b = bf(repr) |
||||
for (x <- self) |
||||
if (p(x)) b ++= f(x).seq |
||||
b.result |
||||
} |
||||
|
||||
/** Applies a function `f` to all elements of the outer $coll containing |
||||
* this `WithFilter` instance that satisfy predicate `p`. |
||||
* |
||||
* @param f the function that is applied for its side-effect to every element. |
||||
* The result of function `f` is discarded. |
||||
* |
||||
* @tparam U the type parameter describing the result of function `f`. |
||||
* This result will always be ignored. Typically `U` is `Unit`, |
||||
* but this is not necessary. |
||||
* |
||||
* @usecase def foreach(f: A => Unit): Unit |
||||
*/ |
||||
def foreach[U](f: A => U): Unit = |
||||
for (x <- self) |
||||
if (p(x)) f(x) |
||||
|
||||
/** Further refines the filter for this $coll. |
||||
* |
||||
* @param q the predicate used to test elements. |
||||
* @return an object of class `WithFilter`, which supports |
||||
* `map`, `flatMap`, `foreach`, and `withFilter` operations. |
||||
* All these operations apply to those elements of this $coll which |
||||
* satisfy the predicate `q` in addition to the predicate `p`. |
||||
*/ |
||||
def withFilter(q: A => Boolean): WithFilter = |
||||
new WithFilter(x => p(x) && q(x)) |
||||
} |
||||
|
||||
// A helper for tails and inits. |
||||
private def iterateUntilEmpty(f: Traversable[A @uV] => Traversable[A @uV]): Iterator[Repr] = { |
||||
val it = Iterator.iterate(thisCollection)(f) takeWhile (x => !x.isEmpty) |
||||
it ++ Iterator(Nil) map (newBuilder ++= _ result) |
||||
} |
||||
} |
||||
|
||||
|
||||
</textarea> |
||||
</form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
theme: "ambiance", |
||||
mode: "text/x-scala" |
||||
}); |
||||
</script> |
||||
</article> |
@ -0,0 +1,55 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "text/x-c"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
MT("indent", |
||||
"[variable-3 void] [def foo]([variable-3 void*] [variable a], [variable-3 int] [variable b]) {", |
||||
" [variable-3 int] [variable c] [operator =] [variable b] [operator +]", |
||||
" [number 1];", |
||||
" [keyword return] [operator *][variable a];", |
||||
"}"); |
||||
|
||||
MT("indent_switch", |
||||
"[keyword switch] ([variable x]) {", |
||||
" [keyword case] [number 10]:", |
||||
" [keyword return] [number 20];", |
||||
" [keyword default]:", |
||||
" [variable printf]([string \"foo %c\"], [variable x]);", |
||||
"}"); |
||||
|
||||
MT("def", |
||||
"[variable-3 void] [def foo]() {}", |
||||
"[keyword struct] [def bar]{}", |
||||
"[variable-3 int] [variable-3 *][def baz]() {}"); |
||||
|
||||
MT("def_new_line", |
||||
"::[variable std]::[variable SomeTerribleType][operator <][variable T][operator >]", |
||||
"[def SomeLongMethodNameThatDoesntFitIntoOneLine]([keyword const] [variable MyType][operator &] [variable param]) {}") |
||||
|
||||
MT("double_block", |
||||
"[keyword for] (;;)", |
||||
" [keyword for] (;;)", |
||||
" [variable x][operator ++];", |
||||
"[keyword return];"); |
||||
|
||||
MT("preprocessor", |
||||
"[meta #define FOO 3]", |
||||
"[variable-3 int] [variable foo];", |
||||
"[meta #define BAR\\]", |
||||
"[meta 4]", |
||||
"[variable-3 unsigned] [variable-3 int] [variable bar] [operator =] [number 8];", |
||||
"[meta #include <baz> ][comment // comment]") |
||||
|
||||
|
||||
var mode_cpp = CodeMirror.getMode({indentUnit: 2}, "text/x-c++src"); |
||||
function MTCPP(name) { test.mode(name, mode_cpp, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
MTCPP("cpp14_literal", |
||||
"[number 10'000];", |
||||
"[number 0b10'000];", |
||||
"[number 0x10'000];", |
||||
"[string '100000'];"); |
||||
})(); |
@ -0,0 +1,306 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/** |
||||
* Author: Hans Engel |
||||
* Branched from CodeMirror's Scheme mode (by Koh Zi Han, based on implementation by Koh Zi Chun) |
||||
*/ |
||||
|
||||
(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("clojure", function (options) { |
||||
var BUILTIN = "builtin", COMMENT = "comment", STRING = "string", CHARACTER = "string-2", |
||||
ATOM = "atom", NUMBER = "number", BRACKET = "bracket", KEYWORD = "keyword", VAR = "variable"; |
||||
var INDENT_WORD_SKIP = options.indentUnit || 2; |
||||
var NORMAL_INDENT_UNIT = options.indentUnit || 2; |
||||
|
||||
function makeKeywords(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
|
||||
var atoms = makeKeywords("true false nil"); |
||||
|
||||
var keywords = makeKeywords( |
||||
"defn defn- def def- defonce defmulti defmethod defmacro defstruct deftype defprotocol defrecord defproject deftest " + |
||||
"slice defalias defhinted defmacro- defn-memo defnk defnk defonce- defunbound defunbound- defvar defvar- let letfn " + |
||||
"do case cond condp for loop recur when when-not when-let when-first if if-let if-not . .. -> ->> doto and or dosync " + |
||||
"doseq dotimes dorun doall load import unimport ns in-ns refer try catch finally throw with-open with-local-vars " + |
||||
"binding gen-class gen-and-load-class gen-and-save-class handler-case handle"); |
||||
|
||||
var builtins = makeKeywords( |
||||
"* *' *1 *2 *3 *agent* *allow-unresolved-vars* *assert* *clojure-version* *command-line-args* *compile-files* " + |
||||
"*compile-path* *compiler-options* *data-readers* *e *err* *file* *flush-on-newline* *fn-loader* *in* " + |
||||
"*math-context* *ns* *out* *print-dup* *print-length* *print-level* *print-meta* *print-readably* *read-eval* " + |
||||
"*source-path* *unchecked-math* *use-context-classloader* *verbose-defrecords* *warn-on-reflection* + +' - -' -> " + |
||||
"->> ->ArrayChunk ->Vec ->VecNode ->VecSeq -cache-protocol-fn -reset-methods .. / < <= = == > >= EMPTY-NODE accessor " + |
||||
"aclone add-classpath add-watch agent agent-error agent-errors aget alength alias all-ns alter alter-meta! " + |
||||
"alter-var-root amap ancestors and apply areduce array-map aset aset-boolean aset-byte aset-char aset-double " + |
||||
"aset-float aset-int aset-long aset-short assert assoc assoc! assoc-in associative? atom await await-for await1 " + |
||||
"bases bean bigdec bigint biginteger binding bit-and bit-and-not bit-clear bit-flip bit-not bit-or bit-set " + |
||||
"bit-shift-left bit-shift-right bit-test bit-xor boolean boolean-array booleans bound-fn bound-fn* bound? butlast " + |
||||
"byte byte-array bytes case cat cast char char-array char-escape-string char-name-string char? chars chunk chunk-append " + |
||||
"chunk-buffer chunk-cons chunk-first chunk-next chunk-rest chunked-seq? class class? clear-agent-errors " + |
||||
"clojure-version coll? comment commute comp comparator compare compare-and-set! compile complement completing concat cond condp " + |
||||
"conj conj! cons constantly construct-proxy contains? count counted? create-ns create-struct cycle dec dec' decimal? " + |
||||
"declare dedupe default-data-readers definline definterface defmacro defmethod defmulti defn defn- defonce defprotocol " + |
||||
"defrecord defstruct deftype delay delay? deliver denominator deref derive descendants destructure disj disj! dissoc " + |
||||
"dissoc! distinct distinct? doall dorun doseq dosync dotimes doto double double-array doubles drop drop-last " + |
||||
"drop-while eduction empty empty? ensure enumeration-seq error-handler error-mode eval even? every-pred every? ex-data ex-info " + |
||||
"extend extend-protocol extend-type extenders extends? false? ffirst file-seq filter filterv find find-keyword " + |
||||
"find-ns find-protocol-impl find-protocol-method find-var first flatten float float-array float? floats flush fn fn? " + |
||||
"fnext fnil for force format frequencies future future-call future-cancel future-cancelled? future-done? future? " + |
||||
"gen-class gen-interface gensym get get-in get-method get-proxy-class get-thread-bindings get-validator group-by hash " + |
||||
"hash-combine hash-map hash-set identical? identity if-let if-not ifn? import in-ns inc inc' init-proxy instance? " + |
||||
"int int-array integer? interleave intern interpose into into-array ints io! isa? iterate iterator-seq juxt keep " + |
||||
"keep-indexed key keys keyword keyword? last lazy-cat lazy-seq let letfn line-seq list list* list? load load-file " + |
||||
"load-reader load-string loaded-libs locking long long-array longs loop macroexpand macroexpand-1 make-array " + |
||||
"make-hierarchy map map-indexed map? mapcat mapv max max-key memfn memoize merge merge-with meta method-sig methods " + |
||||
"min min-key mod munge name namespace namespace-munge neg? newline next nfirst nil? nnext not not-any? not-empty " + |
||||
"not-every? not= ns ns-aliases ns-imports ns-interns ns-map ns-name ns-publics ns-refers ns-resolve ns-unalias " + |
||||
"ns-unmap nth nthnext nthrest num number? numerator object-array odd? or parents partial partition partition-all " + |
||||
"partition-by pcalls peek persistent! pmap pop pop! pop-thread-bindings pos? pr pr-str prefer-method prefers " + |
||||
"primitives-classnames print print-ctor print-dup print-method print-simple print-str printf println println-str " + |
||||
"prn prn-str promise proxy proxy-call-with-super proxy-mappings proxy-name proxy-super push-thread-bindings pvalues " + |
||||
"quot rand rand-int rand-nth random-sample range ratio? rational? rationalize re-find re-groups re-matcher re-matches re-pattern " + |
||||
"re-seq read read-line read-string realized? reduce reduce-kv reductions ref ref-history-count ref-max-history " + |
||||
"ref-min-history ref-set refer refer-clojure reify release-pending-sends rem remove remove-all-methods " + |
||||
"remove-method remove-ns remove-watch repeat repeatedly replace replicate require reset! reset-meta! resolve rest " + |
||||
"restart-agent resultset-seq reverse reversible? rseq rsubseq satisfies? second select-keys send send-off seq seq? " + |
||||
"seque sequence sequential? set set-error-handler! set-error-mode! set-validator! set? short short-array shorts " + |
||||
"shuffle shutdown-agents slurp some some-fn sort sort-by sorted-map sorted-map-by sorted-set sorted-set-by sorted? " + |
||||
"special-symbol? spit split-at split-with str string? struct struct-map subs subseq subvec supers swap! symbol " + |
||||
"symbol? sync take take-last take-nth take-while test the-ns thread-bound? time to-array to-array-2d trampoline transduce " + |
||||
"transient tree-seq true? type unchecked-add unchecked-add-int unchecked-byte unchecked-char unchecked-dec " + |
||||
"unchecked-dec-int unchecked-divide-int unchecked-double unchecked-float unchecked-inc unchecked-inc-int " + |
||||
"unchecked-int unchecked-long unchecked-multiply unchecked-multiply-int unchecked-negate unchecked-negate-int "+ |
||||
"unchecked-remainder-int unchecked-short unchecked-subtract unchecked-subtract-int underive unquote " + |
||||
"unquote-splicing update update-in update-proxy use val vals var-get var-set var? vary-meta vec vector vector-of " + |
||||
"vector? volatile! volatile? vreset! vswap! when when-first when-let when-not while with-bindings with-bindings* with-in-str with-loading-context " + |
||||
"with-local-vars with-meta with-open with-out-str with-precision with-redefs with-redefs-fn xml-seq zero? zipmap " + |
||||
"*default-data-reader-fn* as-> cond-> cond->> reduced reduced? send-via set-agent-send-executor! " + |
||||
"set-agent-send-off-executor! some-> some->>"); |
||||
|
||||
var indentKeys = makeKeywords( |
||||
// Built-ins
|
||||
"ns fn def defn defmethod bound-fn if if-not case condp when while when-not when-first do future comment doto " + |
||||
"locking proxy with-open with-precision reify deftype defrecord defprotocol extend extend-protocol extend-type " + |
||||
"try catch " + |
||||
|
||||
// Binding forms
|
||||
"let letfn binding loop for doseq dotimes when-let if-let " + |
||||
|
||||
// Data structures
|
||||
"defstruct struct-map assoc " + |
||||
|
||||
// clojure.test
|
||||
"testing deftest " + |
||||
|
||||
// contrib
|
||||
"handler-case handle dotrace deftrace"); |
||||
|
||||
var tests = { |
||||
digit: /\d/, |
||||
digit_or_colon: /[\d:]/, |
||||
hex: /[0-9a-f]/i, |
||||
sign: /[+-]/, |
||||
exponent: /e/i, |
||||
keyword_char: /[^\s\(\[\;\)\]]/, |
||||
symbol: /[\w*+!\-\._?:<>\/\xa1-\uffff]/, |
||||
block_indent: /^(?:def|with)[^\/]+$|\/(?:def|with)/ |
||||
}; |
||||
|
||||
function stateStack(indent, type, prev) { // represents a state stack object
|
||||
this.indent = indent; |
||||
this.type = type; |
||||
this.prev = prev; |
||||
} |
||||
|
||||
function pushStack(state, indent, type) { |
||||
state.indentStack = new stateStack(indent, type, state.indentStack); |
||||
} |
||||
|
||||
function popStack(state) { |
||||
state.indentStack = state.indentStack.prev; |
||||
} |
||||
|
||||
function isNumber(ch, stream){ |
||||
// hex
|
||||
if ( ch === '0' && stream.eat(/x/i) ) { |
||||
stream.eatWhile(tests.hex); |
||||
return true; |
||||
} |
||||
|
||||
// leading sign
|
||||
if ( ( ch == '+' || ch == '-' ) && ( tests.digit.test(stream.peek()) ) ) { |
||||
stream.eat(tests.sign); |
||||
ch = stream.next(); |
||||
} |
||||
|
||||
if ( tests.digit.test(ch) ) { |
||||
stream.eat(ch); |
||||
stream.eatWhile(tests.digit); |
||||
|
||||
if ( '.' == stream.peek() ) { |
||||
stream.eat('.'); |
||||
stream.eatWhile(tests.digit); |
||||
} else if ('/' == stream.peek() ) { |
||||
stream.eat('/'); |
||||
stream.eatWhile(tests.digit); |
||||
} |
||||
|
||||
if ( stream.eat(tests.exponent) ) { |
||||
stream.eat(tests.sign); |
||||
stream.eatWhile(tests.digit); |
||||
} |
||||
|
||||
return true; |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
// Eat character that starts after backslash \
|
||||
function eatCharacter(stream) { |
||||
var first = stream.next(); |
||||
// Read special literals: backspace, newline, space, return.
|
||||
// Just read all lowercase letters.
|
||||
if (first && first.match(/[a-z]/) && stream.match(/[a-z]+/, true)) { |
||||
return; |
||||
} |
||||
// Read unicode character: \u1000 \uA0a1
|
||||
if (first === "u") { |
||||
stream.match(/[0-9a-z]{4}/i, true); |
||||
} |
||||
} |
||||
|
||||
return { |
||||
startState: function () { |
||||
return { |
||||
indentStack: null, |
||||
indentation: 0, |
||||
mode: false |
||||
}; |
||||
}, |
||||
|
||||
token: function (stream, state) { |
||||
if (state.indentStack == null && stream.sol()) { |
||||
// update indentation, but only if indentStack is empty
|
||||
state.indentation = stream.indentation(); |
||||
} |
||||
|
||||
// skip spaces
|
||||
if (state.mode != "string" && stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
var returnType = null; |
||||
|
||||
switch(state.mode){ |
||||
case "string": // multi-line string parsing mode
|
||||
var next, escaped = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == "\"" && !escaped) { |
||||
|
||||
state.mode = false; |
||||
break; |
||||
} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
returnType = STRING; // continue on in string mode
|
||||
break; |
||||
default: // default parsing mode
|
||||
var ch = stream.next(); |
||||
|
||||
if (ch == "\"") { |
||||
state.mode = "string"; |
||||
returnType = STRING; |
||||
} else if (ch == "\\") { |
||||
eatCharacter(stream); |
||||
returnType = CHARACTER; |
||||
} else if (ch == "'" && !( tests.digit_or_colon.test(stream.peek()) )) { |
||||
returnType = ATOM; |
||||
} else if (ch == ";") { // comment
|
||||
stream.skipToEnd(); // rest of the line is a comment
|
||||
returnType = COMMENT; |
||||
} else if (isNumber(ch,stream)){ |
||||
returnType = NUMBER; |
||||
} else if (ch == "(" || ch == "[" || ch == "{" ) { |
||||
var keyWord = '', indentTemp = stream.column(), letter; |
||||
/** |
||||
Either |
||||
(indent-word .. |
||||
(non-indent-word .. |
||||
(;something else, bracket, etc. |
||||
*/ |
||||
|
||||
if (ch == "(") while ((letter = stream.eat(tests.keyword_char)) != null) { |
||||
keyWord += letter; |
||||
} |
||||
|
||||
if (keyWord.length > 0 && (indentKeys.propertyIsEnumerable(keyWord) || |
||||
tests.block_indent.test(keyWord))) { // indent-word
|
||||
pushStack(state, indentTemp + INDENT_WORD_SKIP, ch); |
||||
} else { // non-indent word
|
||||
// we continue eating the spaces
|
||||
stream.eatSpace(); |
||||
if (stream.eol() || stream.peek() == ";") { |
||||
// nothing significant after
|
||||
// we restart indentation the user defined spaces after
|
||||
pushStack(state, indentTemp + NORMAL_INDENT_UNIT, ch); |
||||
} else { |
||||
pushStack(state, indentTemp + stream.current().length, ch); // else we match
|
||||
} |
||||
} |
||||
stream.backUp(stream.current().length - 1); // undo all the eating
|
||||
|
||||
returnType = BRACKET; |
||||
} else if (ch == ")" || ch == "]" || ch == "}") { |
||||
returnType = BRACKET; |
||||
if (state.indentStack != null && state.indentStack.type == (ch == ")" ? "(" : (ch == "]" ? "[" :"{"))) { |
||||
popStack(state); |
||||
} |
||||
} else if ( ch == ":" ) { |
||||
stream.eatWhile(tests.symbol); |
||||
return ATOM; |
||||
} else { |
||||
stream.eatWhile(tests.symbol); |
||||
|
||||
if (keywords && keywords.propertyIsEnumerable(stream.current())) { |
||||
returnType = KEYWORD; |
||||
} else if (builtins && builtins.propertyIsEnumerable(stream.current())) { |
||||
returnType = BUILTIN; |
||||
} else if (atoms && atoms.propertyIsEnumerable(stream.current())) { |
||||
returnType = ATOM; |
||||
} else { |
||||
returnType = VAR; |
||||
} |
||||
} |
||||
} |
||||
|
||||
return returnType; |
||||
}, |
||||
|
||||
indent: function (state) { |
||||
if (state.indentStack == null) return state.indentation; |
||||
return state.indentStack.indent; |
||||
}, |
||||
|
||||
closeBrackets: {pairs: "()[]{}\"\""}, |
||||
lineComment: ";;" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-clojure", "clojure"); |
||||
CodeMirror.defineMIME("text/x-clojurescript", "clojure"); |
||||
CodeMirror.defineMIME("application/edn", "clojure"); |
||||
|
||||
}); |
@ -0,0 +1,91 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Clojure mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="clojure.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Clojure</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Clojure mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
; Conway's Game of Life, based on the work of: |
||||
;; Laurent Petit https://gist.github.com/1200343 |
||||
;; Christophe Grand http://clj-me.cgrand.net/2011/08/19/conways-game-of-life |
||||
|
||||
(ns ^{:doc "Conway's Game of Life."} |
||||
game-of-life) |
||||
|
||||
;; Core game of life's algorithm functions |
||||
|
||||
(defn neighbours |
||||
"Given a cell's coordinates, returns the coordinates of its neighbours." |
||||
[[x y]] |
||||
(for [dx [-1 0 1] dy (if (zero? dx) [-1 1] [-1 0 1])] |
||||
[(+ dx x) (+ dy y)])) |
||||
|
||||
(defn step |
||||
"Given a set of living cells, computes the new set of living cells." |
||||
[cells] |
||||
(set (for [[cell n] (frequencies (mapcat neighbours cells)) |
||||
:when (or (= n 3) (and (= n 2) (cells cell)))] |
||||
cell))) |
||||
|
||||
;; Utility methods for displaying game on a text terminal |
||||
|
||||
(defn print-board |
||||
"Prints a board on *out*, representing a step in the game." |
||||
[board w h] |
||||
(doseq [x (range (inc w)) y (range (inc h))] |
||||
(if (= y 0) (print "\n")) |
||||
(print (if (board [x y]) "[X]" " . ")))) |
||||
|
||||
(defn display-grids |
||||
"Prints a squence of boards on *out*, representing several steps." |
||||
[grids w h] |
||||
(doseq [board grids] |
||||
(print-board board w h) |
||||
(print "\n"))) |
||||
|
||||
;; Launches an example board |
||||
|
||||
(def |
||||
^{:doc "board represents the initial set of living cells"} |
||||
board #{[2 1] [2 2] [2 3]}) |
||||
|
||||
(display-grids (take 3 (iterate step board)) 5 5) |
||||
|
||||
;; Let's play with characters |
||||
(println \1 \a \# \\ |
||||
\" \( \newline |
||||
\} \" \space |
||||
\tab \return \backspace |
||||
\u1000 \uAaAa \u9F9F) |
||||
|
||||
;; Let's play with numbers |
||||
(+ 1 -1 1/2 -1/2 -0.5 0.5) |
||||
|
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-clojure</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,97 @@ |
||||
// 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") |
||||
mod(require("../../lib/codemirror")); |
||||
else if (typeof define == "function" && define.amd) |
||||
define(["../../lib/codemirror"], mod); |
||||
else |
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
CodeMirror.defineMode("cmake", function () { |
||||
var variable_regex = /({)?[a-zA-Z0-9_]+(})?/; |
||||
|
||||
function tokenString(stream, state) { |
||||
var current, prev, found_var = false; |
||||
while (!stream.eol() && (current = stream.next()) != state.pending) { |
||||
if (current === '$' && prev != '\\' && state.pending == '"') { |
||||
found_var = true; |
||||
break; |
||||
} |
||||
prev = current; |
||||
} |
||||
if (found_var) { |
||||
stream.backUp(1); |
||||
} |
||||
if (current == state.pending) { |
||||
state.continueString = false; |
||||
} else { |
||||
state.continueString = true; |
||||
} |
||||
return "string"; |
||||
} |
||||
|
||||
function tokenize(stream, state) { |
||||
var ch = stream.next(); |
||||
|
||||
// Have we found a variable?
|
||||
if (ch === '$') { |
||||
if (stream.match(variable_regex)) { |
||||
return 'variable-2'; |
||||
} |
||||
return 'variable'; |
||||
} |
||||
// Should we still be looking for the end of a string?
|
||||
if (state.continueString) { |
||||
// If so, go through the loop again
|
||||
stream.backUp(1); |
||||
return tokenString(stream, state); |
||||
} |
||||
// Do we just have a function on our hands?
|
||||
// In 'cmake_minimum_required (VERSION 2.8.8)', 'cmake_minimum_required' is matched
|
||||
if (stream.match(/(\s+)?\w+\(/) || stream.match(/(\s+)?\w+\ \(/)) { |
||||
stream.backUp(1); |
||||
return 'def'; |
||||
} |
||||
if (ch == "#") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
// Have we found a string?
|
||||
if (ch == "'" || ch == '"') { |
||||
// Store the type (single or double)
|
||||
state.pending = ch; |
||||
// Perform the looping function to find the end
|
||||
return tokenString(stream, state); |
||||
} |
||||
if (ch == '(' || ch == ')') { |
||||
return 'bracket'; |
||||
} |
||||
if (ch.match(/[0-9]/)) { |
||||
return 'number'; |
||||
} |
||||
stream.eatWhile(/[\w-]/); |
||||
return null; |
||||
} |
||||
return { |
||||
startState: function () { |
||||
var state = {}; |
||||
state.inDefinition = false; |
||||
state.inInclude = false; |
||||
state.continueString = false; |
||||
state.pending = false; |
||||
return state; |
||||
}, |
||||
token: function (stream, state) { |
||||
if (stream.eatSpace()) return null; |
||||
return tokenize(stream, state); |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-cmake", "cmake"); |
||||
|
||||
}); |
@ -0,0 +1,129 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: CMake mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="cmake.js"></script> |
||||
<style> |
||||
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;} |
||||
.cm-s-default span.cm-arrow { color: red; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">CMake</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>CMake mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
# vim: syntax=cmake |
||||
if(NOT CMAKE_BUILD_TYPE) |
||||
# default to Release build for GCC builds |
||||
set(CMAKE_BUILD_TYPE Release CACHE STRING |
||||
"Choose the type of build, options are: None(CMAKE_CXX_FLAGS or CMAKE_C_FLAGS used) Debug Release RelWithDebInfo MinSizeRel." |
||||
FORCE) |
||||
endif() |
||||
message(STATUS "cmake version ${CMAKE_VERSION}") |
||||
if(POLICY CMP0025) |
||||
cmake_policy(SET CMP0025 OLD) # report Apple's Clang as just Clang |
||||
endif() |
||||
if(POLICY CMP0042) |
||||
cmake_policy(SET CMP0042 NEW) # MACOSX_RPATH |
||||
endif() |
||||
|
||||
project (x265) |
||||
cmake_minimum_required (VERSION 2.8.8) # OBJECT libraries require 2.8.8 |
||||
include(CheckIncludeFiles) |
||||
include(CheckFunctionExists) |
||||
include(CheckSymbolExists) |
||||
include(CheckCXXCompilerFlag) |
||||
|
||||
# X265_BUILD must be incremented each time the public API is changed |
||||
set(X265_BUILD 48) |
||||
configure_file("${PROJECT_SOURCE_DIR}/x265.def.in" |
||||
"${PROJECT_BINARY_DIR}/x265.def") |
||||
configure_file("${PROJECT_SOURCE_DIR}/x265_config.h.in" |
||||
"${PROJECT_BINARY_DIR}/x265_config.h") |
||||
|
||||
SET(CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/cmake" "${CMAKE_MODULE_PATH}") |
||||
|
||||
# System architecture detection |
||||
string(TOLOWER "${CMAKE_SYSTEM_PROCESSOR}" SYSPROC) |
||||
set(X86_ALIASES x86 i386 i686 x86_64 amd64) |
||||
list(FIND X86_ALIASES "${SYSPROC}" X86MATCH) |
||||
if("${SYSPROC}" STREQUAL "" OR X86MATCH GREATER "-1") |
||||
message(STATUS "Detected x86 target processor") |
||||
set(X86 1) |
||||
add_definitions(-DX265_ARCH_X86=1) |
||||
if("${CMAKE_SIZEOF_VOID_P}" MATCHES 8) |
||||
set(X64 1) |
||||
add_definitions(-DX86_64=1) |
||||
endif() |
||||
elseif(${SYSPROC} STREQUAL "armv6l") |
||||
message(STATUS "Detected ARM target processor") |
||||
set(ARM 1) |
||||
add_definitions(-DX265_ARCH_ARM=1 -DHAVE_ARMV6=1) |
||||
else() |
||||
message(STATUS "CMAKE_SYSTEM_PROCESSOR value `${CMAKE_SYSTEM_PROCESSOR}` is unknown") |
||||
message(STATUS "Please add this value near ${CMAKE_CURRENT_LIST_FILE}:${CMAKE_CURRENT_LIST_LINE}") |
||||
endif() |
||||
|
||||
if(UNIX) |
||||
list(APPEND PLATFORM_LIBS pthread) |
||||
find_library(LIBRT rt) |
||||
if(LIBRT) |
||||
list(APPEND PLATFORM_LIBS rt) |
||||
endif() |
||||
find_package(Numa) |
||||
if(NUMA_FOUND) |
||||
list(APPEND CMAKE_REQUIRED_LIBRARIES ${NUMA_LIBRARY}) |
||||
check_symbol_exists(numa_node_of_cpu numa.h NUMA_V2) |
||||
if(NUMA_V2) |
||||
add_definitions(-DHAVE_LIBNUMA) |
||||
message(STATUS "libnuma found, building with support for NUMA nodes") |
||||
list(APPEND PLATFORM_LIBS ${NUMA_LIBRARY}) |
||||
link_directories(${NUMA_LIBRARY_DIR}) |
||||
include_directories(${NUMA_INCLUDE_DIR}) |
||||
endif() |
||||
endif() |
||||
mark_as_advanced(LIBRT NUMA_FOUND) |
||||
endif(UNIX) |
||||
|
||||
if(X64 AND NOT WIN32) |
||||
option(ENABLE_PIC "Enable Position Independent Code" ON) |
||||
else() |
||||
option(ENABLE_PIC "Enable Position Independent Code" OFF) |
||||
endif(X64 AND NOT WIN32) |
||||
|
||||
# Compiler detection |
||||
if(CMAKE_GENERATOR STREQUAL "Xcode") |
||||
set(XCODE 1) |
||||
endif() |
||||
if (APPLE) |
||||
add_definitions(-DMACOS) |
||||
endif() |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: "text/x-cmake", |
||||
matchBrackets: true, |
||||
indentUnit: 4 |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-cmake</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,255 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/** |
||||
* Author: Gautam Mehta |
||||
* Branched from CodeMirror's Scheme mode |
||||
*/ |
||||
(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("cobol", function () { |
||||
var BUILTIN = "builtin", COMMENT = "comment", STRING = "string", |
||||
ATOM = "atom", NUMBER = "number", KEYWORD = "keyword", MODTAG = "header", |
||||
COBOLLINENUM = "def", PERIOD = "link"; |
||||
function makeKeywords(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
var atoms = makeKeywords("TRUE FALSE ZEROES ZEROS ZERO SPACES SPACE LOW-VALUE LOW-VALUES "); |
||||
var keywords = makeKeywords( |
||||
"ACCEPT ACCESS ACQUIRE ADD ADDRESS " + |
||||
"ADVANCING AFTER ALIAS ALL ALPHABET " + |
||||
"ALPHABETIC ALPHABETIC-LOWER ALPHABETIC-UPPER ALPHANUMERIC ALPHANUMERIC-EDITED " + |
||||
"ALSO ALTER ALTERNATE AND ANY " + |
||||
"ARE AREA AREAS ARITHMETIC ASCENDING " + |
||||
"ASSIGN AT ATTRIBUTE AUTHOR AUTO " + |
||||
"AUTO-SKIP AUTOMATIC B-AND B-EXOR B-LESS " + |
||||
"B-NOT B-OR BACKGROUND-COLOR BACKGROUND-COLOUR BEEP " + |
||||
"BEFORE BELL BINARY BIT BITS " + |
||||
"BLANK BLINK BLOCK BOOLEAN BOTTOM " + |
||||
"BY CALL CANCEL CD CF " + |
||||
"CH CHARACTER CHARACTERS CLASS CLOCK-UNITS " + |
||||
"CLOSE COBOL CODE CODE-SET COL " + |
||||
"COLLATING COLUMN COMMA COMMIT COMMITMENT " + |
||||
"COMMON COMMUNICATION COMP COMP-0 COMP-1 " + |
||||
"COMP-2 COMP-3 COMP-4 COMP-5 COMP-6 " + |
||||
"COMP-7 COMP-8 COMP-9 COMPUTATIONAL COMPUTATIONAL-0 " + |
||||
"COMPUTATIONAL-1 COMPUTATIONAL-2 COMPUTATIONAL-3 COMPUTATIONAL-4 COMPUTATIONAL-5 " + |
||||
"COMPUTATIONAL-6 COMPUTATIONAL-7 COMPUTATIONAL-8 COMPUTATIONAL-9 COMPUTE " + |
||||
"CONFIGURATION CONNECT CONSOLE CONTAINED CONTAINS " + |
||||
"CONTENT CONTINUE CONTROL CONTROL-AREA CONTROLS " + |
||||
"CONVERTING COPY CORR CORRESPONDING COUNT " + |
||||
"CRT CRT-UNDER CURRENCY CURRENT CURSOR " + |
||||
"DATA DATE DATE-COMPILED DATE-WRITTEN DAY " + |
||||
"DAY-OF-WEEK DB DB-ACCESS-CONTROL-KEY DB-DATA-NAME DB-EXCEPTION " + |
||||
"DB-FORMAT-NAME DB-RECORD-NAME DB-SET-NAME DB-STATUS DBCS " + |
||||
"DBCS-EDITED DE DEBUG-CONTENTS DEBUG-ITEM DEBUG-LINE " + |
||||
"DEBUG-NAME DEBUG-SUB-1 DEBUG-SUB-2 DEBUG-SUB-3 DEBUGGING " + |
||||
"DECIMAL-POINT DECLARATIVES DEFAULT DELETE DELIMITED " + |
||||
"DELIMITER DEPENDING DESCENDING DESCRIBED DESTINATION " + |
||||
"DETAIL DISABLE DISCONNECT DISPLAY DISPLAY-1 " + |
||||
"DISPLAY-2 DISPLAY-3 DISPLAY-4 DISPLAY-5 DISPLAY-6 " + |
||||
"DISPLAY-7 DISPLAY-8 DISPLAY-9 DIVIDE DIVISION " + |
||||
"DOWN DROP DUPLICATE DUPLICATES DYNAMIC " + |
||||
"EBCDIC EGI EJECT ELSE EMI " + |
||||
"EMPTY EMPTY-CHECK ENABLE END END. END-ACCEPT END-ACCEPT. " + |
||||
"END-ADD END-CALL END-COMPUTE END-DELETE END-DISPLAY " + |
||||
"END-DIVIDE END-EVALUATE END-IF END-INVOKE END-MULTIPLY " + |
||||
"END-OF-PAGE END-PERFORM END-READ END-RECEIVE END-RETURN " + |
||||
"END-REWRITE END-SEARCH END-START END-STRING END-SUBTRACT " + |
||||
"END-UNSTRING END-WRITE END-XML ENTER ENTRY " + |
||||
"ENVIRONMENT EOP EQUAL EQUALS ERASE " + |
||||
"ERROR ESI EVALUATE EVERY EXCEEDS " + |
||||
"EXCEPTION EXCLUSIVE EXIT EXTEND EXTERNAL " + |
||||
"EXTERNALLY-DESCRIBED-KEY FD FETCH FILE FILE-CONTROL " + |
||||
"FILE-STREAM FILES FILLER FINAL FIND " + |
||||
"FINISH FIRST FOOTING FOR FOREGROUND-COLOR " + |
||||
"FOREGROUND-COLOUR FORMAT FREE FROM FULL " + |
||||
"FUNCTION GENERATE GET GIVING GLOBAL " + |
||||
"GO GOBACK GREATER GROUP HEADING " + |
||||
"HIGH-VALUE HIGH-VALUES HIGHLIGHT I-O I-O-CONTROL " + |
||||
"ID IDENTIFICATION IF IN INDEX " + |
||||
"INDEX-1 INDEX-2 INDEX-3 INDEX-4 INDEX-5 " + |
||||
"INDEX-6 INDEX-7 INDEX-8 INDEX-9 INDEXED " + |
||||
"INDIC INDICATE INDICATOR INDICATORS INITIAL " + |
||||
"INITIALIZE INITIATE INPUT INPUT-OUTPUT INSPECT " + |
||||
"INSTALLATION INTO INVALID INVOKE IS " + |
||||
"JUST JUSTIFIED KANJI KEEP KEY " + |
||||
"LABEL LAST LD LEADING LEFT " + |
||||
"LEFT-JUSTIFY LENGTH LENGTH-CHECK LESS LIBRARY " + |
||||
"LIKE LIMIT LIMITS LINAGE LINAGE-COUNTER " + |
||||
"LINE LINE-COUNTER LINES LINKAGE LOCAL-STORAGE " + |
||||
"LOCALE LOCALLY LOCK " + |
||||
"MEMBER MEMORY MERGE MESSAGE METACLASS " + |
||||
"MODE MODIFIED MODIFY MODULES MOVE " + |
||||
"MULTIPLE MULTIPLY NATIONAL NATIVE NEGATIVE " + |
||||
"NEXT NO NO-ECHO NONE NOT " + |
||||
"NULL NULL-KEY-MAP NULL-MAP NULLS NUMBER " + |
||||
"NUMERIC NUMERIC-EDITED OBJECT OBJECT-COMPUTER OCCURS " + |
||||
"OF OFF OMITTED ON ONLY " + |
||||
"OPEN OPTIONAL OR ORDER ORGANIZATION " + |
||||
"OTHER OUTPUT OVERFLOW OWNER PACKED-DECIMAL " + |
||||
"PADDING PAGE PAGE-COUNTER PARSE PERFORM " + |
||||
"PF PH PIC PICTURE PLUS " + |
||||
"POINTER POSITION POSITIVE PREFIX PRESENT " + |
||||
"PRINTING PRIOR PROCEDURE PROCEDURE-POINTER PROCEDURES " + |
||||
"PROCEED PROCESS PROCESSING PROGRAM PROGRAM-ID " + |
||||
"PROMPT PROTECTED PURGE QUEUE QUOTE " + |
||||
"QUOTES RANDOM RD READ READY " + |
||||
"REALM RECEIVE RECONNECT RECORD RECORD-NAME " + |
||||
"RECORDS RECURSIVE REDEFINES REEL REFERENCE " + |
||||
"REFERENCE-MONITOR REFERENCES RELATION RELATIVE RELEASE " + |
||||
"REMAINDER REMOVAL RENAMES REPEATED REPLACE " + |
||||
"REPLACING REPORT REPORTING REPORTS REPOSITORY " + |
||||
"REQUIRED RERUN RESERVE RESET RETAINING " + |
||||
"RETRIEVAL RETURN RETURN-CODE RETURNING REVERSE-VIDEO " + |
||||
"REVERSED REWIND REWRITE RF RH " + |
||||
"RIGHT RIGHT-JUSTIFY ROLLBACK ROLLING ROUNDED " + |
||||
"RUN SAME SCREEN SD SEARCH " + |
||||
"SECTION SECURE SECURITY SEGMENT SEGMENT-LIMIT " + |
||||
"SELECT SEND SENTENCE SEPARATE SEQUENCE " + |
||||
"SEQUENTIAL SET SHARED SIGN SIZE " + |
||||
"SKIP1 SKIP2 SKIP3 SORT SORT-MERGE " + |
||||
"SORT-RETURN SOURCE SOURCE-COMPUTER SPACE-FILL " + |
||||
"SPECIAL-NAMES STANDARD STANDARD-1 STANDARD-2 " + |
||||
"START STARTING STATUS STOP STORE " + |
||||
"STRING SUB-QUEUE-1 SUB-QUEUE-2 SUB-QUEUE-3 SUB-SCHEMA " + |
||||
"SUBFILE SUBSTITUTE SUBTRACT SUM SUPPRESS " + |
||||
"SYMBOLIC SYNC SYNCHRONIZED SYSIN SYSOUT " + |
||||
"TABLE TALLYING TAPE TENANT TERMINAL " + |
||||
"TERMINATE TEST TEXT THAN THEN " + |
||||
"THROUGH THRU TIME TIMES TITLE " + |
||||
"TO TOP TRAILING TRAILING-SIGN TRANSACTION " + |
||||
"TYPE TYPEDEF UNDERLINE UNEQUAL UNIT " + |
||||
"UNSTRING UNTIL UP UPDATE UPON " + |
||||
"USAGE USAGE-MODE USE USING VALID " + |
||||
"VALIDATE VALUE VALUES VARYING VLR " + |
||||
"WAIT WHEN WHEN-COMPILED WITH WITHIN " + |
||||
"WORDS WORKING-STORAGE WRITE XML XML-CODE " + |
||||
"XML-EVENT XML-NTEXT XML-TEXT ZERO ZERO-FILL " ); |
||||
|
||||
var builtins = makeKeywords("- * ** / + < <= = > >= "); |
||||
var tests = { |
||||
digit: /\d/, |
||||
digit_or_colon: /[\d:]/, |
||||
hex: /[0-9a-f]/i, |
||||
sign: /[+-]/, |
||||
exponent: /e/i, |
||||
keyword_char: /[^\s\(\[\;\)\]]/, |
||||
symbol: /[\w*+\-]/ |
||||
}; |
||||
function isNumber(ch, stream){ |
||||
// hex
|
||||
if ( ch === '0' && stream.eat(/x/i) ) { |
||||
stream.eatWhile(tests.hex); |
||||
return true; |
||||
} |
||||
// leading sign
|
||||
if ( ( ch == '+' || ch == '-' ) && ( tests.digit.test(stream.peek()) ) ) { |
||||
stream.eat(tests.sign); |
||||
ch = stream.next(); |
||||
} |
||||
if ( tests.digit.test(ch) ) { |
||||
stream.eat(ch); |
||||
stream.eatWhile(tests.digit); |
||||
if ( '.' == stream.peek()) { |
||||
stream.eat('.'); |
||||
stream.eatWhile(tests.digit); |
||||
} |
||||
if ( stream.eat(tests.exponent) ) { |
||||
stream.eat(tests.sign); |
||||
stream.eatWhile(tests.digit); |
||||
} |
||||
return true; |
||||
} |
||||
return false; |
||||
} |
||||
return { |
||||
startState: function () { |
||||
return { |
||||
indentStack: null, |
||||
indentation: 0, |
||||
mode: false |
||||
}; |
||||
}, |
||||
token: function (stream, state) { |
||||
if (state.indentStack == null && stream.sol()) { |
||||
// update indentation, but only if indentStack is empty
|
||||
state.indentation = 6 ; //stream.indentation();
|
||||
} |
||||
// skip spaces
|
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
var returnType = null; |
||||
switch(state.mode){ |
||||
case "string": // multi-line string parsing mode
|
||||
var next = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == "\"" || next == "\'") { |
||||
state.mode = false; |
||||
break; |
||||
} |
||||
} |
||||
returnType = STRING; // continue on in string mode
|
||||
break; |
||||
default: // default parsing mode
|
||||
var ch = stream.next(); |
||||
var col = stream.column(); |
||||
if (col >= 0 && col <= 5) { |
||||
returnType = COBOLLINENUM; |
||||
} else if (col >= 72 && col <= 79) { |
||||
stream.skipToEnd(); |
||||
returnType = MODTAG; |
||||
} else if (ch == "*" && col == 6) { // comment
|
||||
stream.skipToEnd(); // rest of the line is a comment
|
||||
returnType = COMMENT; |
||||
} else if (ch == "\"" || ch == "\'") { |
||||
state.mode = "string"; |
||||
returnType = STRING; |
||||
} else if (ch == "'" && !( tests.digit_or_colon.test(stream.peek()) )) { |
||||
returnType = ATOM; |
||||
} else if (ch == ".") { |
||||
returnType = PERIOD; |
||||
} else if (isNumber(ch,stream)){ |
||||
returnType = NUMBER; |
||||
} else { |
||||
if (stream.current().match(tests.symbol)) { |
||||
while (col < 71) { |
||||
if (stream.eat(tests.symbol) === undefined) { |
||||
break; |
||||
} else { |
||||
col++; |
||||
} |
||||
} |
||||
} |
||||
if (keywords && keywords.propertyIsEnumerable(stream.current().toUpperCase())) { |
||||
returnType = KEYWORD; |
||||
} else if (builtins && builtins.propertyIsEnumerable(stream.current().toUpperCase())) { |
||||
returnType = BUILTIN; |
||||
} else if (atoms && atoms.propertyIsEnumerable(stream.current().toUpperCase())) { |
||||
returnType = ATOM; |
||||
} else returnType = null; |
||||
} |
||||
} |
||||
return returnType; |
||||
}, |
||||
indent: function (state) { |
||||
if (state.indentStack == null) return state.indentation; |
||||
return state.indentStack.indent; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-cobol", "cobol"); |
||||
|
||||
}); |
@ -0,0 +1,210 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: COBOL mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/neat.css"> |
||||
<link rel="stylesheet" href="../../theme/elegant.css"> |
||||
<link rel="stylesheet" href="../../theme/erlang-dark.css"> |
||||
<link rel="stylesheet" href="../../theme/night.css"> |
||||
<link rel="stylesheet" href="../../theme/monokai.css"> |
||||
<link rel="stylesheet" href="../../theme/cobalt.css"> |
||||
<link rel="stylesheet" href="../../theme/eclipse.css"> |
||||
<link rel="stylesheet" href="../../theme/rubyblue.css"> |
||||
<link rel="stylesheet" href="../../theme/lesser-dark.css"> |
||||
<link rel="stylesheet" href="../../theme/xq-dark.css"> |
||||
<link rel="stylesheet" href="../../theme/xq-light.css"> |
||||
<link rel="stylesheet" href="../../theme/ambiance.css"> |
||||
<link rel="stylesheet" href="../../theme/blackboard.css"> |
||||
<link rel="stylesheet" href="../../theme/vibrant-ink.css"> |
||||
<link rel="stylesheet" href="../../theme/solarized.css"> |
||||
<link rel="stylesheet" href="../../theme/twilight.css"> |
||||
<link rel="stylesheet" href="../../theme/midnight.css"> |
||||
<link rel="stylesheet" href="../../addon/dialog/dialog.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="cobol.js"></script> |
||||
<script src="../../addon/selection/active-line.js"></script> |
||||
<script src="../../addon/search/search.js"></script> |
||||
<script src="../../addon/dialog/dialog.js"></script> |
||||
<script src="../../addon/search/searchcursor.js"></script> |
||||
<style> |
||||
.CodeMirror { |
||||
border: 1px solid #eee; |
||||
font-size : 20px; |
||||
height : auto !important; |
||||
} |
||||
.CodeMirror-activeline-background {background: #555555 !important;} |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">COBOL</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>COBOL mode</h2> |
||||
|
||||
<p> Select Theme <select onchange="selectTheme()" id="selectTheme"> |
||||
<option>default</option> |
||||
<option>ambiance</option> |
||||
<option>blackboard</option> |
||||
<option>cobalt</option> |
||||
<option>eclipse</option> |
||||
<option>elegant</option> |
||||
<option>erlang-dark</option> |
||||
<option>lesser-dark</option> |
||||
<option>midnight</option> |
||||
<option>monokai</option> |
||||
<option>neat</option> |
||||
<option>night</option> |
||||
<option>rubyblue</option> |
||||
<option>solarized dark</option> |
||||
<option>solarized light</option> |
||||
<option selected>twilight</option> |
||||
<option>vibrant-ink</option> |
||||
<option>xq-dark</option> |
||||
<option>xq-light</option> |
||||
</select> Select Font Size <select onchange="selectFontsize()" id="selectFontSize"> |
||||
<option value="13px">13px</option> |
||||
<option value="14px">14px</option> |
||||
<option value="16px">16px</option> |
||||
<option value="18px">18px</option> |
||||
<option value="20px" selected="selected">20px</option> |
||||
<option value="24px">24px</option> |
||||
<option value="26px">26px</option> |
||||
<option value="28px">28px</option> |
||||
<option value="30px">30px</option> |
||||
<option value="32px">32px</option> |
||||
<option value="34px">34px</option> |
||||
<option value="36px">36px</option> |
||||
</select> |
||||
<label for="checkBoxReadOnly">Read-only</label> |
||||
<input type="checkbox" id="checkBoxReadOnly" onchange="selectReadOnly()"> |
||||
<label for="id_tabToIndentSpace">Insert Spaces on Tab</label> |
||||
<input type="checkbox" id="id_tabToIndentSpace" onchange="tabToIndentSpace()"> |
||||
</p> |
||||
<textarea id="code" name="code"> |
||||
---------1---------2---------3---------4---------5---------6---------7---------8 |
||||
12345678911234567892123456789312345678941234567895123456789612345678971234567898 |
||||
000010 IDENTIFICATION DIVISION. MODTGHERE |
||||
000020 PROGRAM-ID. SAMPLE. |
||||
000030 AUTHOR. TEST SAM. |
||||
000040 DATE-WRITTEN. 5 February 2013 |
||||
000041 |
||||
000042* A sample program just to show the form. |
||||
000043* The program copies its input to the output, |
||||
000044* and counts the number of records. |
||||
000045* At the end this number is printed. |
||||
000046 |
||||
000050 ENVIRONMENT DIVISION. |
||||
000060 INPUT-OUTPUT SECTION. |
||||
000070 FILE-CONTROL. |
||||
000080 SELECT STUDENT-FILE ASSIGN TO SYSIN |
||||
000090 ORGANIZATION IS LINE SEQUENTIAL. |
||||
000100 SELECT PRINT-FILE ASSIGN TO SYSOUT |
||||
000110 ORGANIZATION IS LINE SEQUENTIAL. |
||||
000120 |
||||
000130 DATA DIVISION. |
||||
000140 FILE SECTION. |
||||
000150 FD STUDENT-FILE |
||||
000160 RECORD CONTAINS 43 CHARACTERS |
||||
000170 DATA RECORD IS STUDENT-IN. |
||||
000180 01 STUDENT-IN PIC X(43). |
||||
000190 |
||||
000200 FD PRINT-FILE |
||||
000210 RECORD CONTAINS 80 CHARACTERS |
||||
000220 DATA RECORD IS PRINT-LINE. |
||||
000230 01 PRINT-LINE PIC X(80). |
||||
000240 |
||||
000250 WORKING-STORAGE SECTION. |
||||
000260 01 DATA-REMAINS-SWITCH PIC X(2) VALUE SPACES. |
||||
000261 01 RECORDS-WRITTEN PIC 99. |
||||
000270 |
||||
000280 01 DETAIL-LINE. |
||||
000290 05 FILLER PIC X(7) VALUE SPACES. |
||||
000300 05 RECORD-IMAGE PIC X(43). |
||||
000310 05 FILLER PIC X(30) VALUE SPACES. |
||||
000311 |
||||
000312 01 SUMMARY-LINE. |
||||
000313 05 FILLER PIC X(7) VALUE SPACES. |
||||
000314 05 TOTAL-READ PIC 99. |
||||
000315 05 FILLER PIC X VALUE SPACE. |
||||
000316 05 FILLER PIC X(17) |
||||
000317 VALUE 'Records were read'. |
||||
000318 05 FILLER PIC X(53) VALUE SPACES. |
||||
000319 |
||||
000320 PROCEDURE DIVISION. |
||||
000321 |
||||
000330 PREPARE-SENIOR-REPORT. |
||||
000340 OPEN INPUT STUDENT-FILE |
||||
000350 OUTPUT PRINT-FILE. |
||||
000351 MOVE ZERO TO RECORDS-WRITTEN. |
||||
000360 READ STUDENT-FILE |
||||
000370 AT END MOVE 'NO' TO DATA-REMAINS-SWITCH |
||||
000380 END-READ. |
||||
000390 PERFORM PROCESS-RECORDS |
||||
000410 UNTIL DATA-REMAINS-SWITCH = 'NO'. |
||||
000411 PERFORM PRINT-SUMMARY. |
||||
000420 CLOSE STUDENT-FILE |
||||
000430 PRINT-FILE. |
||||
000440 STOP RUN. |
||||
000450 |
||||
000460 PROCESS-RECORDS. |
||||
000470 MOVE STUDENT-IN TO RECORD-IMAGE. |
||||
000480 MOVE DETAIL-LINE TO PRINT-LINE. |
||||
000490 WRITE PRINT-LINE. |
||||
000500 ADD 1 TO RECORDS-WRITTEN. |
||||
000510 READ STUDENT-FILE |
||||
000520 AT END MOVE 'NO' TO DATA-REMAINS-SWITCH |
||||
000530 END-READ. |
||||
000540 |
||||
000550 PRINT-SUMMARY. |
||||
000560 MOVE RECORDS-WRITTEN TO TOTAL-READ. |
||||
000570 MOVE SUMMARY-LINE TO PRINT-LINE. |
||||
000571 WRITE PRINT-LINE. |
||||
000572 |
||||
000580 |
||||
</textarea> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-cobol", |
||||
theme : "twilight", |
||||
styleActiveLine: true, |
||||
showCursorWhenSelecting : true, |
||||
}); |
||||
function selectTheme() { |
||||
var themeInput = document.getElementById("selectTheme"); |
||||
var theme = themeInput.options[themeInput.selectedIndex].innerHTML; |
||||
editor.setOption("theme", theme); |
||||
} |
||||
function selectFontsize() { |
||||
var fontSizeInput = document.getElementById("selectFontSize"); |
||||
var fontSize = fontSizeInput.options[fontSizeInput.selectedIndex].innerHTML; |
||||
editor.getWrapperElement().style.fontSize = fontSize; |
||||
editor.refresh(); |
||||
} |
||||
function selectReadOnly() { |
||||
editor.setOption("readOnly", document.getElementById("checkBoxReadOnly").checked); |
||||
} |
||||
function tabToIndentSpace() { |
||||
if (document.getElementById("id_tabToIndentSpace").checked) { |
||||
editor.setOption("extraKeys", {Tab: function(cm) { cm.replaceSelection(" ", "end"); }}); |
||||
} else { |
||||
editor.setOption("extraKeys", {Tab: function(cm) { cm.replaceSelection(" ", "end"); }}); |
||||
} |
||||
} |
||||
</script> |
||||
</article> |
@ -0,0 +1,355 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/** |
||||
* Link to the project's GitHub page: |
||||
* https://github.com/pickhardt/coffeescript-codemirror-mode
|
||||
*/ |
||||
(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("coffeescript", function(conf, parserConf) { |
||||
var ERRORCLASS = "error"; |
||||
|
||||
function wordRegexp(words) { |
||||
return new RegExp("^((" + words.join(")|(") + "))\\b"); |
||||
} |
||||
|
||||
var operators = /^(?:->|=>|\+[+=]?|-[\-=]?|\*[\*=]?|\/[\/=]?|[=!]=|<[><]?=?|>>?=?|%=?|&=?|\|=?|\^=?|\~|!|\?|(or|and|\|\||&&|\?)=)/; |
||||
var delimiters = /^(?:[()\[\]{},:`=;]|\.\.?\.?)/; |
||||
var identifiers = /^[_A-Za-z$][_A-Za-z$0-9]*/; |
||||
var atProp = /^@[_A-Za-z$][_A-Za-z$0-9]*/; |
||||
|
||||
var wordOperators = wordRegexp(["and", "or", "not", |
||||
"is", "isnt", "in", |
||||
"instanceof", "typeof"]); |
||||
var indentKeywords = ["for", "while", "loop", "if", "unless", "else", |
||||
"switch", "try", "catch", "finally", "class"]; |
||||
var commonKeywords = ["break", "by", "continue", "debugger", "delete", |
||||
"do", "in", "of", "new", "return", "then", |
||||
"this", "@", "throw", "when", "until", "extends"]; |
||||
|
||||
var keywords = wordRegexp(indentKeywords.concat(commonKeywords)); |
||||
|
||||
indentKeywords = wordRegexp(indentKeywords); |
||||
|
||||
|
||||
var stringPrefixes = /^('{3}|\"{3}|['\"])/; |
||||
var regexPrefixes = /^(\/{3}|\/)/; |
||||
var commonConstants = ["Infinity", "NaN", "undefined", "null", "true", "false", "on", "off", "yes", "no"]; |
||||
var constants = wordRegexp(commonConstants); |
||||
|
||||
// Tokenizers
|
||||
function tokenBase(stream, state) { |
||||
// Handle scope changes
|
||||
if (stream.sol()) { |
||||
if (state.scope.align === null) state.scope.align = false; |
||||
var scopeOffset = state.scope.offset; |
||||
if (stream.eatSpace()) { |
||||
var lineOffset = stream.indentation(); |
||||
if (lineOffset > scopeOffset && state.scope.type == "coffee") { |
||||
return "indent"; |
||||
} else if (lineOffset < scopeOffset) { |
||||
return "dedent"; |
||||
} |
||||
return null; |
||||
} else { |
||||
if (scopeOffset > 0) { |
||||
dedent(stream, state); |
||||
} |
||||
} |
||||
} |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
var ch = stream.peek(); |
||||
|
||||
// Handle docco title comment (single line)
|
||||
if (stream.match("####")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
|
||||
// Handle multi line comments
|
||||
if (stream.match("###")) { |
||||
state.tokenize = longComment; |
||||
return state.tokenize(stream, state); |
||||
} |
||||
|
||||
// Single line comment
|
||||
if (ch === "#") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
|
||||
// Handle number literals
|
||||
if (stream.match(/^-?[0-9\.]/, false)) { |
||||
var floatLiteral = false; |
||||
// Floats
|
||||
if (stream.match(/^-?\d*\.\d+(e[\+\-]?\d+)?/i)) { |
||||
floatLiteral = true; |
||||
} |
||||
if (stream.match(/^-?\d+\.\d*/)) { |
||||
floatLiteral = true; |
||||
} |
||||
if (stream.match(/^-?\.\d+/)) { |
||||
floatLiteral = true; |
||||
} |
||||
|
||||
if (floatLiteral) { |
||||
// prevent from getting extra . on 1..
|
||||
if (stream.peek() == "."){ |
||||
stream.backUp(1); |
||||
} |
||||
return "number"; |
||||
} |
||||
// Integers
|
||||
var intLiteral = false; |
||||
// Hex
|
||||
if (stream.match(/^-?0x[0-9a-f]+/i)) { |
||||
intLiteral = true; |
||||
} |
||||
// Decimal
|
||||
if (stream.match(/^-?[1-9]\d*(e[\+\-]?\d+)?/)) { |
||||
intLiteral = true; |
||||
} |
||||
// Zero by itself with no other piece of number.
|
||||
if (stream.match(/^-?0(?![\dx])/i)) { |
||||
intLiteral = true; |
||||
} |
||||
if (intLiteral) { |
||||
return "number"; |
||||
} |
||||
} |
||||
|
||||
// Handle strings
|
||||
if (stream.match(stringPrefixes)) { |
||||
state.tokenize = tokenFactory(stream.current(), false, "string"); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
// Handle regex literals
|
||||
if (stream.match(regexPrefixes)) { |
||||
if (stream.current() != "/" || stream.match(/^.*\//, false)) { // prevent highlight of division
|
||||
state.tokenize = tokenFactory(stream.current(), true, "string-2"); |
||||
return state.tokenize(stream, state); |
||||
} else { |
||||
stream.backUp(1); |
||||
} |
||||
} |
||||
|
||||
|
||||
|
||||
// Handle operators and delimiters
|
||||
if (stream.match(operators) || stream.match(wordOperators)) { |
||||
return "operator"; |
||||
} |
||||
if (stream.match(delimiters)) { |
||||
return "punctuation"; |
||||
} |
||||
|
||||
if (stream.match(constants)) { |
||||
return "atom"; |
||||
} |
||||
|
||||
if (stream.match(atProp) || state.prop && stream.match(identifiers)) { |
||||
return "property"; |
||||
} |
||||
|
||||
if (stream.match(keywords)) { |
||||
return "keyword"; |
||||
} |
||||
|
||||
if (stream.match(identifiers)) { |
||||
return "variable"; |
||||
} |
||||
|
||||
// Handle non-detected items
|
||||
stream.next(); |
||||
return ERRORCLASS; |
||||
} |
||||
|
||||
function tokenFactory(delimiter, singleline, outclass) { |
||||
return function(stream, state) { |
||||
while (!stream.eol()) { |
||||
stream.eatWhile(/[^'"\/\\]/); |
||||
if (stream.eat("\\")) { |
||||
stream.next(); |
||||
if (singleline && stream.eol()) { |
||||
return outclass; |
||||
} |
||||
} else if (stream.match(delimiter)) { |
||||
state.tokenize = tokenBase; |
||||
return outclass; |
||||
} else { |
||||
stream.eat(/['"\/]/); |
||||
} |
||||
} |
||||
if (singleline) { |
||||
if (parserConf.singleLineStringErrors) { |
||||
outclass = ERRORCLASS; |
||||
} else { |
||||
state.tokenize = tokenBase; |
||||
} |
||||
} |
||||
return outclass; |
||||
}; |
||||
} |
||||
|
||||
function longComment(stream, state) { |
||||
while (!stream.eol()) { |
||||
stream.eatWhile(/[^#]/); |
||||
if (stream.match("###")) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
stream.eatWhile("#"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
function indent(stream, state, type) { |
||||
type = type || "coffee"; |
||||
var offset = 0, align = false, alignOffset = null; |
||||
for (var scope = state.scope; scope; scope = scope.prev) { |
||||
if (scope.type === "coffee" || scope.type == "}") { |
||||
offset = scope.offset + conf.indentUnit; |
||||
break; |
||||
} |
||||
} |
||||
if (type !== "coffee") { |
||||
align = null; |
||||
alignOffset = stream.column() + stream.current().length; |
||||
} else if (state.scope.align) { |
||||
state.scope.align = false; |
||||
} |
||||
state.scope = { |
||||
offset: offset, |
||||
type: type, |
||||
prev: state.scope, |
||||
align: align, |
||||
alignOffset: alignOffset |
||||
}; |
||||
} |
||||
|
||||
function dedent(stream, state) { |
||||
if (!state.scope.prev) return; |
||||
if (state.scope.type === "coffee") { |
||||
var _indent = stream.indentation(); |
||||
var matched = false; |
||||
for (var scope = state.scope; scope; scope = scope.prev) { |
||||
if (_indent === scope.offset) { |
||||
matched = true; |
||||
break; |
||||
} |
||||
} |
||||
if (!matched) { |
||||
return true; |
||||
} |
||||
while (state.scope.prev && state.scope.offset !== _indent) { |
||||
state.scope = state.scope.prev; |
||||
} |
||||
return false; |
||||
} else { |
||||
state.scope = state.scope.prev; |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
function tokenLexer(stream, state) { |
||||
var style = state.tokenize(stream, state); |
||||
var current = stream.current(); |
||||
|
||||
// Handle scope changes.
|
||||
if (current === "return") { |
||||
state.dedent = true; |
||||
} |
||||
if (((current === "->" || current === "=>") && stream.eol()) |
||||
|| style === "indent") { |
||||
indent(stream, state); |
||||
} |
||||
var delimiter_index = "[({".indexOf(current); |
||||
if (delimiter_index !== -1) { |
||||
indent(stream, state, "])}".slice(delimiter_index, delimiter_index+1)); |
||||
} |
||||
if (indentKeywords.exec(current)){ |
||||
indent(stream, state); |
||||
} |
||||
if (current == "then"){ |
||||
dedent(stream, state); |
||||
} |
||||
|
||||
|
||||
if (style === "dedent") { |
||||
if (dedent(stream, state)) { |
||||
return ERRORCLASS; |
||||
} |
||||
} |
||||
delimiter_index = "])}".indexOf(current); |
||||
if (delimiter_index !== -1) { |
||||
while (state.scope.type == "coffee" && state.scope.prev) |
||||
state.scope = state.scope.prev; |
||||
if (state.scope.type == current) |
||||
state.scope = state.scope.prev; |
||||
} |
||||
if (state.dedent && stream.eol()) { |
||||
if (state.scope.type == "coffee" && state.scope.prev) |
||||
state.scope = state.scope.prev; |
||||
state.dedent = false; |
||||
} |
||||
|
||||
return style; |
||||
} |
||||
|
||||
var external = { |
||||
startState: function(basecolumn) { |
||||
return { |
||||
tokenize: tokenBase, |
||||
scope: {offset:basecolumn || 0, type:"coffee", prev: null, align: false}, |
||||
prop: false, |
||||
dedent: 0 |
||||
}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
var fillAlign = state.scope.align === null && state.scope; |
||||
if (fillAlign && stream.sol()) fillAlign.align = false; |
||||
|
||||
var style = tokenLexer(stream, state); |
||||
if (style && style != "comment") { |
||||
if (fillAlign) fillAlign.align = true; |
||||
state.prop = style == "punctuation" && stream.current() == "." |
||||
} |
||||
|
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, text) { |
||||
if (state.tokenize != tokenBase) return 0; |
||||
var scope = state.scope; |
||||
var closer = text && "])}".indexOf(text.charAt(0)) > -1; |
||||
if (closer) while (scope.type == "coffee" && scope.prev) scope = scope.prev; |
||||
var closes = closer && scope.type === text.charAt(0); |
||||
if (scope.align) |
||||
return scope.alignOffset - (closes ? 1 : 0); |
||||
else |
||||
return (closes ? scope.prev : scope).offset; |
||||
}, |
||||
|
||||
lineComment: "#", |
||||
fold: "indent" |
||||
}; |
||||
return external; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-coffeescript", "coffeescript"); |
||||
CodeMirror.defineMIME("text/coffeescript", "coffeescript"); |
||||
|
||||
}); |
@ -0,0 +1,123 @@ |
||||
// 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("commonlisp", function (config) { |
||||
var specialForm = /^(block|let*|return-from|catch|load-time-value|setq|eval-when|locally|symbol-macrolet|flet|macrolet|tagbody|function|multiple-value-call|the|go|multiple-value-prog1|throw|if|progn|unwind-protect|labels|progv|let|quote)$/; |
||||
var assumeBody = /^with|^def|^do|^prog|case$|^cond$|bind$|when$|unless$/; |
||||
var numLiteral = /^(?:[+\-]?(?:\d+|\d*\.\d+)(?:[efd][+\-]?\d+)?|[+\-]?\d+(?:\/[+\-]?\d+)?|#b[+\-]?[01]+|#o[+\-]?[0-7]+|#x[+\-]?[\da-f]+)/; |
||||
var symbol = /[^\s'`,@()\[\]";]/; |
||||
var type; |
||||
|
||||
function readSym(stream) { |
||||
var ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "\\") stream.next(); |
||||
else if (!symbol.test(ch)) { stream.backUp(1); break; } |
||||
} |
||||
return stream.current(); |
||||
} |
||||
|
||||
function base(stream, state) { |
||||
if (stream.eatSpace()) {type = "ws"; return null;} |
||||
if (stream.match(numLiteral)) return "number"; |
||||
var ch = stream.next(); |
||||
if (ch == "\\") ch = stream.next(); |
||||
|
||||
if (ch == '"') return (state.tokenize = inString)(stream, state); |
||||
else if (ch == "(") { type = "open"; return "bracket"; } |
||||
else if (ch == ")" || ch == "]") { type = "close"; return "bracket"; } |
||||
else if (ch == ";") { stream.skipToEnd(); type = "ws"; return "comment"; } |
||||
else if (/['`,@]/.test(ch)) return null; |
||||
else if (ch == "|") { |
||||
if (stream.skipTo("|")) { stream.next(); return "symbol"; } |
||||
else { stream.skipToEnd(); return "error"; } |
||||
} else if (ch == "#") { |
||||
var ch = stream.next(); |
||||
if (ch == "[") { type = "open"; return "bracket"; } |
||||
else if (/[+\-=\.']/.test(ch)) return null; |
||||
else if (/\d/.test(ch) && stream.match(/^\d*#/)) return null; |
||||
else if (ch == "|") return (state.tokenize = inComment)(stream, state); |
||||
else if (ch == ":") { readSym(stream); return "meta"; } |
||||
else return "error"; |
||||
} else { |
||||
var name = readSym(stream); |
||||
if (name == ".") return null; |
||||
type = "symbol"; |
||||
if (name == "nil" || name == "t" || name.charAt(0) == ":") return "atom"; |
||||
if (state.lastType == "open" && (specialForm.test(name) || assumeBody.test(name))) return "keyword"; |
||||
if (name.charAt(0) == "&") return "variable-2"; |
||||
return "variable"; |
||||
} |
||||
} |
||||
|
||||
function inString(stream, state) { |
||||
var escaped = false, next; |
||||
while (next = stream.next()) { |
||||
if (next == '"' && !escaped) { state.tokenize = base; break; } |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
return "string"; |
||||
} |
||||
|
||||
function inComment(stream, state) { |
||||
var next, last; |
||||
while (next = stream.next()) { |
||||
if (next == "#" && last == "|") { state.tokenize = base; break; } |
||||
last = next; |
||||
} |
||||
type = "ws"; |
||||
return "comment"; |
||||
} |
||||
|
||||
return { |
||||
startState: function () { |
||||
return {ctx: {prev: null, start: 0, indentTo: 0}, lastType: null, tokenize: base}; |
||||
}, |
||||
|
||||
token: function (stream, state) { |
||||
if (stream.sol() && typeof state.ctx.indentTo != "number") |
||||
state.ctx.indentTo = state.ctx.start + 1; |
||||
|
||||
type = null; |
||||
var style = state.tokenize(stream, state); |
||||
if (type != "ws") { |
||||
if (state.ctx.indentTo == null) { |
||||
if (type == "symbol" && assumeBody.test(stream.current())) |
||||
state.ctx.indentTo = state.ctx.start + config.indentUnit; |
||||
else |
||||
state.ctx.indentTo = "next"; |
||||
} else if (state.ctx.indentTo == "next") { |
||||
state.ctx.indentTo = stream.column(); |
||||
} |
||||
state.lastType = type; |
||||
} |
||||
if (type == "open") state.ctx = {prev: state.ctx, start: stream.column(), indentTo: null}; |
||||
else if (type == "close") state.ctx = state.ctx.prev || state.ctx; |
||||
return style; |
||||
}, |
||||
|
||||
indent: function (state, _textAfter) { |
||||
var i = state.ctx.indentTo; |
||||
return typeof i == "number" ? i : state.ctx.start + 1; |
||||
}, |
||||
|
||||
closeBrackets: {pairs: "()[]{}\"\""}, |
||||
lineComment: ";;", |
||||
blockCommentStart: "#|", |
||||
blockCommentEnd: "|#" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-common-lisp", "commonlisp"); |
||||
|
||||
}); |
@ -0,0 +1,177 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Common Lisp mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="commonlisp.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Common Lisp</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Common Lisp mode</h2> |
||||
<form><textarea id="code" name="code">(in-package :cl-postgres) |
||||
|
||||
;; These are used to synthesize reader and writer names for integer |
||||
;; reading/writing functions when the amount of bytes and the |
||||
;; signedness is known. Both the macro that creates the functions and |
||||
;; some macros that use them create names this way. |
||||
(eval-when (:compile-toplevel :load-toplevel :execute) |
||||
(defun integer-reader-name (bytes signed) |
||||
(intern (with-standard-io-syntax |
||||
(format nil "~a~a~a~a" '#:read- (if signed "" '#:u) '#:int bytes)))) |
||||
(defun integer-writer-name (bytes signed) |
||||
(intern (with-standard-io-syntax |
||||
(format nil "~a~a~a~a" '#:write- (if signed "" '#:u) '#:int bytes))))) |
||||
|
||||
(defmacro integer-reader (bytes) |
||||
"Create a function to read integers from a binary stream." |
||||
(let ((bits (* bytes 8))) |
||||
(labels ((return-form (signed) |
||||
(if signed |
||||
`(if (logbitp ,(1- bits) result) |
||||
(dpb result (byte ,(1- bits) 0) -1) |
||||
result) |
||||
`result)) |
||||
(generate-reader (signed) |
||||
`(defun ,(integer-reader-name bytes signed) (socket) |
||||
(declare (type stream socket) |
||||
#.*optimize*) |
||||
,(if (= bytes 1) |
||||
`(let ((result (the (unsigned-byte 8) (read-byte socket)))) |
||||
(declare (type (unsigned-byte 8) result)) |
||||
,(return-form signed)) |
||||
`(let ((result 0)) |
||||
(declare (type (unsigned-byte ,bits) result)) |
||||
,@(loop :for byte :from (1- bytes) :downto 0 |
||||
:collect `(setf (ldb (byte 8 ,(* 8 byte)) result) |
||||
(the (unsigned-byte 8) (read-byte socket)))) |
||||
,(return-form signed)))))) |
||||
`(progn |
||||
;; This causes weird errors on SBCL in some circumstances. Disabled for now. |
||||
;; (declaim (inline ,(integer-reader-name bytes t) |
||||
;; ,(integer-reader-name bytes nil))) |
||||
(declaim (ftype (function (t) (signed-byte ,bits)) |
||||
,(integer-reader-name bytes t))) |
||||
,(generate-reader t) |
||||
(declaim (ftype (function (t) (unsigned-byte ,bits)) |
||||
,(integer-reader-name bytes nil))) |
||||
,(generate-reader nil))))) |
||||
|
||||
(defmacro integer-writer (bytes) |
||||
"Create a function to write integers to a binary stream." |
||||
(let ((bits (* 8 bytes))) |
||||
`(progn |
||||
(declaim (inline ,(integer-writer-name bytes t) |
||||
,(integer-writer-name bytes nil))) |
||||
(defun ,(integer-writer-name bytes nil) (socket value) |
||||
(declare (type stream socket) |
||||
(type (unsigned-byte ,bits) value) |
||||
#.*optimize*) |
||||
,@(if (= bytes 1) |
||||
`((write-byte value socket)) |
||||
(loop :for byte :from (1- bytes) :downto 0 |
||||
:collect `(write-byte (ldb (byte 8 ,(* byte 8)) value) |
||||
socket))) |
||||
(values)) |
||||
(defun ,(integer-writer-name bytes t) (socket value) |
||||
(declare (type stream socket) |
||||
(type (signed-byte ,bits) value) |
||||
#.*optimize*) |
||||
,@(if (= bytes 1) |
||||
`((write-byte (ldb (byte 8 0) value) socket)) |
||||
(loop :for byte :from (1- bytes) :downto 0 |
||||
:collect `(write-byte (ldb (byte 8 ,(* byte 8)) value) |
||||
socket))) |
||||
(values))))) |
||||
|
||||
;; All the instances of the above that we need. |
||||
|
||||
(integer-reader 1) |
||||
(integer-reader 2) |
||||
(integer-reader 4) |
||||
(integer-reader 8) |
||||
|
||||
(integer-writer 1) |
||||
(integer-writer 2) |
||||
(integer-writer 4) |
||||
|
||||
(defun write-bytes (socket bytes) |
||||
"Write a byte-array to a stream." |
||||
(declare (type stream socket) |
||||
(type (simple-array (unsigned-byte 8)) bytes) |
||||
#.*optimize*) |
||||
(write-sequence bytes socket)) |
||||
|
||||
(defun write-str (socket string) |
||||
"Write a null-terminated string to a stream \(encoding it when UTF-8 |
||||
support is enabled.)." |
||||
(declare (type stream socket) |
||||
(type string string) |
||||
#.*optimize*) |
||||
(enc-write-string string socket) |
||||
(write-uint1 socket 0)) |
||||
|
||||
(declaim (ftype (function (t unsigned-byte) |
||||
(simple-array (unsigned-byte 8) (*))) |
||||
read-bytes)) |
||||
(defun read-bytes (socket length) |
||||
"Read a byte array of the given length from a stream." |
||||
(declare (type stream socket) |
||||
(type fixnum length) |
||||
#.*optimize*) |
||||
(let ((result (make-array length :element-type '(unsigned-byte 8)))) |
||||
(read-sequence result socket) |
||||
result)) |
||||
|
||||
(declaim (ftype (function (t) string) read-str)) |
||||
(defun read-str (socket) |
||||
"Read a null-terminated string from a stream. Takes care of encoding |
||||
when UTF-8 support is enabled." |
||||
(declare (type stream socket) |
||||
#.*optimize*) |
||||
(enc-read-string socket :null-terminated t)) |
||||
|
||||
(defun skip-bytes (socket length) |
||||
"Skip a given number of bytes in a binary stream." |
||||
(declare (type stream socket) |
||||
(type (unsigned-byte 32) length) |
||||
#.*optimize*) |
||||
(dotimes (i length) |
||||
(read-byte socket))) |
||||
|
||||
(defun skip-str (socket) |
||||
"Skip a null-terminated string." |
||||
(declare (type stream socket) |
||||
#.*optimize*) |
||||
(loop :for char :of-type fixnum = (read-byte socket) |
||||
:until (zerop char))) |
||||
|
||||
(defun ensure-socket-is-closed (socket &key abort) |
||||
(when (open-stream-p socket) |
||||
(handler-case |
||||
(close socket :abort abort) |
||||
(error (error) |
||||
(warn "Ignoring the error which happened while trying to close PostgreSQL socket: ~A" error))))) |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {lineNumbers: true}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-common-lisp</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,391 @@ |
||||
// 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("crystal", function(config) { |
||||
function wordRegExp(words, end) { |
||||
return new RegExp((end ? "" : "^") + "(?:" + words.join("|") + ")" + (end ? "$" : "\\b")); |
||||
} |
||||
|
||||
function chain(tokenize, stream, state) { |
||||
state.tokenize.push(tokenize); |
||||
return tokenize(stream, state); |
||||
} |
||||
|
||||
var operators = /^(?:[-+/%|&^]|\*\*?|[<>]{2})/; |
||||
var conditionalOperators = /^(?:[=!]~|===|<=>|[<>=!]=?|[|&]{2}|~)/; |
||||
var indexingOperators = /^(?:\[\][?=]?)/; |
||||
var anotherOperators = /^(?:\.(?:\.{2})?|->|[?:])/; |
||||
var idents = /^[a-z_\u009F-\uFFFF][a-zA-Z0-9_\u009F-\uFFFF]*/; |
||||
var types = /^[A-Z_\u009F-\uFFFF][a-zA-Z0-9_\u009F-\uFFFF]*/; |
||||
var keywords = wordRegExp([ |
||||
"abstract", "alias", "as", "asm", "begin", "break", "case", "class", "def", "do", |
||||
"else", "elsif", "end", "ensure", "enum", "extend", "for", "fun", "if", "ifdef", |
||||
"include", "instance_sizeof", "lib", "macro", "module", "next", "of", "out", "pointerof", |
||||
"private", "protected", "rescue", "return", "require", "sizeof", "struct", |
||||
"super", "then", "type", "typeof", "union", "unless", "until", "when", "while", "with", |
||||
"yield", "__DIR__", "__FILE__", "__LINE__" |
||||
]); |
||||
var atomWords = wordRegExp(["true", "false", "nil", "self"]); |
||||
var indentKeywordsArray = [ |
||||
"def", "fun", "macro", |
||||
"class", "module", "struct", "lib", "enum", "union", |
||||
"if", "unless", "case", "while", "until", "begin", "then", |
||||
"do", |
||||
"for", "ifdef" |
||||
]; |
||||
var indentKeywords = wordRegExp(indentKeywordsArray); |
||||
var dedentKeywordsArray = [ |
||||
"end", |
||||
"else", "elsif", |
||||
"rescue", "ensure" |
||||
]; |
||||
var dedentKeywords = wordRegExp(dedentKeywordsArray); |
||||
var dedentPunctualsArray = ["\\)", "\\}", "\\]"]; |
||||
var dedentPunctuals = new RegExp("^(?:" + dedentPunctualsArray.join("|") + ")$"); |
||||
var nextTokenizer = { |
||||
"def": tokenFollowIdent, "fun": tokenFollowIdent, "macro": tokenMacroDef, |
||||
"class": tokenFollowType, "module": tokenFollowType, "struct": tokenFollowType, |
||||
"lib": tokenFollowType, "enum": tokenFollowType, "union": tokenFollowType |
||||
}; |
||||
var matching = {"[": "]", "{": "}", "(": ")", "<": ">"}; |
||||
|
||||
function tokenBase(stream, state) { |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
// Macros
|
||||
if (state.lastToken != "\\" && stream.match("{%", false)) { |
||||
return chain(tokenMacro("%", "%"), stream, state); |
||||
} |
||||
|
||||
if (state.lastToken != "\\" && stream.match("{{", false)) { |
||||
return chain(tokenMacro("{", "}"), stream, state); |
||||
} |
||||
|
||||
// Comments
|
||||
if (stream.peek() == "#") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
|
||||
// Variables and keywords
|
||||
var matched; |
||||
if (stream.match(idents)) { |
||||
stream.eat(/[?!]/); |
||||
|
||||
matched = stream.current(); |
||||
if (stream.eat(":")) { |
||||
return "atom"; |
||||
} else if (state.lastToken == ".") { |
||||
return "property"; |
||||
} else if (keywords.test(matched)) { |
||||
if (state.lastToken != "abstract" && indentKeywords.test(matched)) { |
||||
if (!(matched == "fun" && state.blocks.indexOf("lib") >= 0)) { |
||||
state.blocks.push(matched); |
||||
state.currentIndent += 1; |
||||
} |
||||
} else if (dedentKeywords.test(matched)) { |
||||
state.blocks.pop(); |
||||
state.currentIndent -= 1; |
||||
} |
||||
|
||||
if (nextTokenizer.hasOwnProperty(matched)) { |
||||
state.tokenize.push(nextTokenizer[matched]); |
||||
} |
||||
|
||||
return "keyword"; |
||||
} else if (atomWords.test(matched)) { |
||||
return "atom"; |
||||
} |
||||
|
||||
return "variable"; |
||||
} |
||||
|
||||
// Class variables and instance variables
|
||||
// or attributes
|
||||
if (stream.eat("@")) { |
||||
if (stream.peek() == "[") { |
||||
return chain(tokenNest("[", "]", "meta"), stream, state); |
||||
} |
||||
|
||||
stream.eat("@"); |
||||
stream.match(idents) || stream.match(types); |
||||
return "variable-2"; |
||||
} |
||||
|
||||
// Global variables
|
||||
if (stream.eat("$")) { |
||||
stream.eat(/[0-9]+|\?/) || stream.match(idents) || stream.match(types); |
||||
return "variable-3"; |
||||
} |
||||
|
||||
// Constants and types
|
||||
if (stream.match(types)) { |
||||
return "tag"; |
||||
} |
||||
|
||||
// Symbols or ':' operator
|
||||
if (stream.eat(":")) { |
||||
if (stream.eat("\"")) { |
||||
return chain(tokenQuote("\"", "atom", false), stream, state); |
||||
} else if (stream.match(idents) || stream.match(types) || |
||||
stream.match(operators) || stream.match(conditionalOperators) || stream.match(indexingOperators)) { |
||||
return "atom"; |
||||
} |
||||
stream.eat(":"); |
||||
return "operator"; |
||||
} |
||||
|
||||
// Strings
|
||||
if (stream.eat("\"")) { |
||||
return chain(tokenQuote("\"", "string", true), stream, state); |
||||
} |
||||
|
||||
// Strings or regexps or macro variables or '%' operator
|
||||
if (stream.peek() == "%") { |
||||
var style = "string"; |
||||
var embed = true; |
||||
var delim; |
||||
|
||||
if (stream.match("%r")) { |
||||
// Regexps
|
||||
style = "string-2"; |
||||
delim = stream.next(); |
||||
} else if (stream.match("%w")) { |
||||
embed = false; |
||||
delim = stream.next(); |
||||
} else { |
||||
if(delim = stream.match(/^%([^\w\s=])/)) { |
||||
delim = delim[1]; |
||||
} else if (stream.match(/^%[a-zA-Z0-9_\u009F-\uFFFF]*/)) { |
||||
// Macro variables
|
||||
return "meta"; |
||||
} else { |
||||
// '%' operator
|
||||
return "operator"; |
||||
} |
||||
} |
||||
|
||||
if (matching.hasOwnProperty(delim)) { |
||||
delim = matching[delim]; |
||||
} |
||||
return chain(tokenQuote(delim, style, embed), stream, state); |
||||
} |
||||
|
||||
// Characters
|
||||
if (stream.eat("'")) { |
||||
stream.match(/^(?:[^']|\\(?:[befnrtv0'"]|[0-7]{3}|u(?:[0-9a-fA-F]{4}|\{[0-9a-fA-F]{1,6}\})))/); |
||||
stream.eat("'"); |
||||
return "atom"; |
||||
} |
||||
|
||||
// Numbers
|
||||
if (stream.eat("0")) { |
||||
if (stream.eat("x")) { |
||||
stream.match(/^[0-9a-fA-F]+/); |
||||
} else if (stream.eat("o")) { |
||||
stream.match(/^[0-7]+/); |
||||
} else if (stream.eat("b")) { |
||||
stream.match(/^[01]+/); |
||||
} |
||||
return "number"; |
||||
} |
||||
|
||||
if (stream.eat(/\d/)) { |
||||
stream.match(/^\d*(?:\.\d+)?(?:[eE][+-]?\d+)?/); |
||||
return "number"; |
||||
} |
||||
|
||||
// Operators
|
||||
if (stream.match(operators)) { |
||||
stream.eat("="); // Operators can follow assign symbol.
|
||||
return "operator"; |
||||
} |
||||
|
||||
if (stream.match(conditionalOperators) || stream.match(anotherOperators)) { |
||||
return "operator"; |
||||
} |
||||
|
||||
// Parens and braces
|
||||
if (matched = stream.match(/[({[]/, false)) { |
||||
matched = matched[0]; |
||||
return chain(tokenNest(matched, matching[matched], null), stream, state); |
||||
} |
||||
|
||||
// Escapes
|
||||
if (stream.eat("\\")) { |
||||
stream.next(); |
||||
return "meta"; |
||||
} |
||||
|
||||
stream.next(); |
||||
return null; |
||||
} |
||||
|
||||
function tokenNest(begin, end, style, started) { |
||||
return function (stream, state) { |
||||
if (!started && stream.match(begin)) { |
||||
state.tokenize[state.tokenize.length - 1] = tokenNest(begin, end, style, true); |
||||
state.currentIndent += 1; |
||||
return style; |
||||
} |
||||
|
||||
var nextStyle = tokenBase(stream, state); |
||||
if (stream.current() === end) { |
||||
state.tokenize.pop(); |
||||
state.currentIndent -= 1; |
||||
nextStyle = style; |
||||
} |
||||
|
||||
return nextStyle; |
||||
}; |
||||
} |
||||
|
||||
function tokenMacro(begin, end, started) { |
||||
return function (stream, state) { |
||||
if (!started && stream.match("{" + begin)) { |
||||
state.currentIndent += 1; |
||||
state.tokenize[state.tokenize.length - 1] = tokenMacro(begin, end, true); |
||||
return "meta"; |
||||
} |
||||
|
||||
if (stream.match(end + "}")) { |
||||
state.currentIndent -= 1; |
||||
state.tokenize.pop(); |
||||
return "meta"; |
||||
} |
||||
|
||||
return tokenBase(stream, state); |
||||
}; |
||||
} |
||||
|
||||
function tokenMacroDef(stream, state) { |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
var matched; |
||||
if (matched = stream.match(idents)) { |
||||
if (matched == "def") { |
||||
return "keyword"; |
||||
} |
||||
stream.eat(/[?!]/); |
||||
} |
||||
|
||||
state.tokenize.pop(); |
||||
return "def"; |
||||
} |
||||
|
||||
function tokenFollowIdent(stream, state) { |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
if (stream.match(idents)) { |
||||
stream.eat(/[!?]/); |
||||
} else { |
||||
stream.match(operators) || stream.match(conditionalOperators) || stream.match(indexingOperators); |
||||
} |
||||
state.tokenize.pop(); |
||||
return "def"; |
||||
} |
||||
|
||||
function tokenFollowType(stream, state) { |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
stream.match(types); |
||||
state.tokenize.pop(); |
||||
return "def"; |
||||
} |
||||
|
||||
function tokenQuote(end, style, embed) { |
||||
return function (stream, state) { |
||||
var escaped = false; |
||||
|
||||
while (stream.peek()) { |
||||
if (!escaped) { |
||||
if (stream.match("{%", false)) { |
||||
state.tokenize.push(tokenMacro("%", "%")); |
||||
return style; |
||||
} |
||||
|
||||
if (stream.match("{{", false)) { |
||||
state.tokenize.push(tokenMacro("{", "}")); |
||||
return style; |
||||
} |
||||
|
||||
if (embed && stream.match("#{", false)) { |
||||
state.tokenize.push(tokenNest("#{", "}", "meta")); |
||||
return style; |
||||
} |
||||
|
||||
var ch = stream.next(); |
||||
|
||||
if (ch == end) { |
||||
state.tokenize.pop(); |
||||
return style; |
||||
} |
||||
|
||||
escaped = ch == "\\"; |
||||
} else { |
||||
stream.next(); |
||||
escaped = false; |
||||
} |
||||
} |
||||
|
||||
return style; |
||||
}; |
||||
} |
||||
|
||||
return { |
||||
startState: function () { |
||||
return { |
||||
tokenize: [tokenBase], |
||||
currentIndent: 0, |
||||
lastToken: null, |
||||
blocks: [] |
||||
}; |
||||
}, |
||||
|
||||
token: function (stream, state) { |
||||
var style = state.tokenize[state.tokenize.length - 1](stream, state); |
||||
var token = stream.current(); |
||||
|
||||
if (style && style != "comment") { |
||||
state.lastToken = token; |
||||
} |
||||
|
||||
return style; |
||||
}, |
||||
|
||||
indent: function (state, textAfter) { |
||||
textAfter = textAfter.replace(/^\s*(?:\{%)?\s*|\s*(?:%\})?\s*$/g, ""); |
||||
|
||||
if (dedentKeywords.test(textAfter) || dedentPunctuals.test(textAfter)) { |
||||
return config.indentUnit * (state.currentIndent - 1); |
||||
} |
||||
|
||||
return config.indentUnit * state.currentIndent; |
||||
}, |
||||
|
||||
fold: "indent", |
||||
electricInput: wordRegExp(dedentPunctualsArray.concat(dedentKeywordsArray), true), |
||||
lineComment: '#' |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-crystal", "crystal"); |
||||
}); |
@ -0,0 +1,119 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Crystal mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="crystal.js"></script> |
||||
<style> |
||||
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;} |
||||
.cm-s-default span.cm-arrow { color: red; } |
||||
</style> |
||||
|
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Crystal</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Crystal mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
# Features of Crystal |
||||
# - Ruby-inspired syntax. |
||||
# - Statically type-checked but without having to specify the type of variables or method arguments. |
||||
# - Be able to call C code by writing bindings to it in Crystal. |
||||
# - Have compile-time evaluation and generation of code, to avoid boilerplate code. |
||||
# - Compile to efficient native code. |
||||
|
||||
# A very basic HTTP server |
||||
require "http/server" |
||||
|
||||
server = HTTP::Server.new(8080) do |request| |
||||
HTTP::Response.ok "text/plain", "Hello world, got #{request.path}!" |
||||
end |
||||
|
||||
puts "Listening on http://0.0.0.0:8080" |
||||
server.listen |
||||
|
||||
module Foo |
||||
def initialize(@foo); end |
||||
|
||||
abstract def abstract_method : String |
||||
|
||||
@[AlwaysInline] |
||||
def with_foofoo |
||||
with Foo.new(self) yield |
||||
end |
||||
|
||||
struct Foo |
||||
def initialize(@foo); end |
||||
|
||||
def hello_world |
||||
@foo.abstract_method |
||||
end |
||||
end |
||||
end |
||||
|
||||
class Bar |
||||
include Foo |
||||
|
||||
@@foobar = 12345 |
||||
|
||||
def initialize(@bar) |
||||
super(@bar.not_nil! + 100) |
||||
end |
||||
|
||||
macro alias_method(name, method) |
||||
def {{ name }}(*args) |
||||
{{ method }}(*args) |
||||
end |
||||
end |
||||
|
||||
def a_method |
||||
"Hello, World" |
||||
end |
||||
|
||||
alias_method abstract_method, a_method |
||||
|
||||
macro def show_instance_vars : Nil |
||||
{% for var in @type.instance_vars %} |
||||
puts "@{{ var }} = #{ @{{ var }} }" |
||||
{% end %} |
||||
nil |
||||
end |
||||
end |
||||
|
||||
class Baz < Bar; end |
||||
|
||||
lib LibC |
||||
fun c_puts = "puts"(str : Char*) : Int |
||||
end |
||||
|
||||
$baz = Baz.new(100) |
||||
$baz.show_instance_vars |
||||
$baz.with_foofoo do |
||||
LibC.c_puts hello_world |
||||
end |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: "text/x-crystal", |
||||
matchBrackets: true, |
||||
indentUnit: 2 |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-crystal</code>.</p> |
||||
</article> |
@ -0,0 +1,825 @@ |
||||
// 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("css", function(config, parserConfig) { |
||||
var inline = parserConfig.inline |
||||
if (!parserConfig.propertyKeywords) parserConfig = CodeMirror.resolveMode("text/css"); |
||||
|
||||
var indentUnit = config.indentUnit, |
||||
tokenHooks = parserConfig.tokenHooks, |
||||
documentTypes = parserConfig.documentTypes || {}, |
||||
mediaTypes = parserConfig.mediaTypes || {}, |
||||
mediaFeatures = parserConfig.mediaFeatures || {}, |
||||
mediaValueKeywords = parserConfig.mediaValueKeywords || {}, |
||||
propertyKeywords = parserConfig.propertyKeywords || {}, |
||||
nonStandardPropertyKeywords = parserConfig.nonStandardPropertyKeywords || {}, |
||||
fontProperties = parserConfig.fontProperties || {}, |
||||
counterDescriptors = parserConfig.counterDescriptors || {}, |
||||
colorKeywords = parserConfig.colorKeywords || {}, |
||||
valueKeywords = parserConfig.valueKeywords || {}, |
||||
allowNested = parserConfig.allowNested, |
||||
supportsAtComponent = parserConfig.supportsAtComponent === true; |
||||
|
||||
var type, override; |
||||
function ret(style, tp) { type = tp; return style; } |
||||
|
||||
// Tokenizers
|
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (tokenHooks[ch]) { |
||||
var result = tokenHooks[ch](stream, state); |
||||
if (result !== false) return result; |
||||
} |
||||
if (ch == "@") { |
||||
stream.eatWhile(/[\w\\\-]/); |
||||
return ret("def", stream.current()); |
||||
} else if (ch == "=" || (ch == "~" || ch == "|") && stream.eat("=")) { |
||||
return ret(null, "compare"); |
||||
} else if (ch == "\"" || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} else if (ch == "#") { |
||||
stream.eatWhile(/[\w\\\-]/); |
||||
return ret("atom", "hash"); |
||||
} else if (ch == "!") { |
||||
stream.match(/^\s*\w*/); |
||||
return ret("keyword", "important"); |
||||
} else if (/\d/.test(ch) || ch == "." && stream.eat(/\d/)) { |
||||
stream.eatWhile(/[\w.%]/); |
||||
return ret("number", "unit"); |
||||
} else if (ch === "-") { |
||||
if (/[\d.]/.test(stream.peek())) { |
||||
stream.eatWhile(/[\w.%]/); |
||||
return ret("number", "unit"); |
||||
} else if (stream.match(/^-[\w\\\-]+/)) { |
||||
stream.eatWhile(/[\w\\\-]/); |
||||
if (stream.match(/^\s*:/, false)) |
||||
return ret("variable-2", "variable-definition"); |
||||
return ret("variable-2", "variable"); |
||||
} else if (stream.match(/^\w+-/)) { |
||||
return ret("meta", "meta"); |
||||
} |
||||
} else if (/[,+>*\/]/.test(ch)) { |
||||
return ret(null, "select-op"); |
||||
} else if (ch == "." && stream.match(/^-?[_a-z][_a-z0-9-]*/i)) { |
||||
return ret("qualifier", "qualifier"); |
||||
} else if (/[:;{}\[\]\(\)]/.test(ch)) { |
||||
return ret(null, ch); |
||||
} else if ((ch == "u" && stream.match(/rl(-prefix)?\(/)) || |
||||
(ch == "d" && stream.match("omain(")) || |
||||
(ch == "r" && stream.match("egexp("))) { |
||||
stream.backUp(1); |
||||
state.tokenize = tokenParenthesized; |
||||
return ret("property", "word"); |
||||
} else if (/[\w\\\-]/.test(ch)) { |
||||
stream.eatWhile(/[\w\\\-]/); |
||||
return ret("property", "word"); |
||||
} else { |
||||
return ret(null, null); |
||||
} |
||||
} |
||||
|
||||
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 ret("string", "string"); |
||||
}; |
||||
} |
||||
|
||||
function tokenParenthesized(stream, state) { |
||||
stream.next(); // Must be '('
|
||||
if (!stream.match(/\s*[\"\')]/, false)) |
||||
state.tokenize = tokenString(")"); |
||||
else |
||||
state.tokenize = null; |
||||
return ret(null, "("); |
||||
} |
||||
|
||||
// Context management
|
||||
|
||||
function Context(type, indent, prev) { |
||||
this.type = type; |
||||
this.indent = indent; |
||||
this.prev = prev; |
||||
} |
||||
|
||||
function pushContext(state, stream, type, indent) { |
||||
state.context = new Context(type, stream.indentation() + (indent === false ? 0 : indentUnit), state.context); |
||||
return type; |
||||
} |
||||
|
||||
function popContext(state) { |
||||
if (state.context.prev) |
||||
state.context = state.context.prev; |
||||
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 wordAsValue(stream) { |
||||
var word = stream.current().toLowerCase(); |
||||
if (valueKeywords.hasOwnProperty(word)) |
||||
override = "atom"; |
||||
else if (colorKeywords.hasOwnProperty(word)) |
||||
override = "keyword"; |
||||
else |
||||
override = "variable"; |
||||
} |
||||
|
||||
var states = {}; |
||||
|
||||
states.top = function(type, stream, state) { |
||||
if (type == "{") { |
||||
return pushContext(state, stream, "block"); |
||||
} else if (type == "}" && state.context.prev) { |
||||
return popContext(state); |
||||
} else if (supportsAtComponent && /@component/.test(type)) { |
||||
return pushContext(state, stream, "atComponentBlock"); |
||||
} else if (/^@(-moz-)?document$/.test(type)) { |
||||
return pushContext(state, stream, "documentTypes"); |
||||
} else if (/^@(media|supports|(-moz-)?document|import)$/.test(type)) { |
||||
return pushContext(state, stream, "atBlock"); |
||||
} else if (/^@(font-face|counter-style)/.test(type)) { |
||||
state.stateArg = type; |
||||
return "restricted_atBlock_before"; |
||||
} else if (/^@(-(moz|ms|o|webkit)-)?keyframes$/.test(type)) { |
||||
return "keyframes"; |
||||
} else if (type && type.charAt(0) == "@") { |
||||
return pushContext(state, stream, "at"); |
||||
} else if (type == "hash") { |
||||
override = "builtin"; |
||||
} else if (type == "word") { |
||||
override = "tag"; |
||||
} else if (type == "variable-definition") { |
||||
return "maybeprop"; |
||||
} else if (type == "interpolation") { |
||||
return pushContext(state, stream, "interpolation"); |
||||
} else if (type == ":") { |
||||
return "pseudo"; |
||||
} else if (allowNested && type == "(") { |
||||
return pushContext(state, stream, "parens"); |
||||
} |
||||
return state.context.type; |
||||
}; |
||||
|
||||
states.block = function(type, stream, state) { |
||||
if (type == "word") { |
||||
var word = stream.current().toLowerCase(); |
||||
if (propertyKeywords.hasOwnProperty(word)) { |
||||
override = "property"; |
||||
return "maybeprop"; |
||||
} else if (nonStandardPropertyKeywords.hasOwnProperty(word)) { |
||||
override = "string-2"; |
||||
return "maybeprop"; |
||||
} else if (allowNested) { |
||||
override = stream.match(/^\s*:(?:\s|$)/, false) ? "property" : "tag"; |
||||
return "block"; |
||||
} else { |
||||
override += " error"; |
||||
return "maybeprop"; |
||||
} |
||||
} else if (type == "meta") { |
||||
return "block"; |
||||
} else if (!allowNested && (type == "hash" || type == "qualifier")) { |
||||
override = "error"; |
||||
return "block"; |
||||
} else { |
||||
return states.top(type, stream, state); |
||||
} |
||||
}; |
||||
|
||||
states.maybeprop = function(type, stream, state) { |
||||
if (type == ":") return pushContext(state, stream, "prop"); |
||||
return pass(type, stream, state); |
||||
}; |
||||
|
||||
states.prop = function(type, stream, state) { |
||||
if (type == ";") return popContext(state); |
||||
if (type == "{" && allowNested) return pushContext(state, stream, "propBlock"); |
||||
if (type == "}" || type == "{") return popAndPass(type, stream, state); |
||||
if (type == "(") return pushContext(state, stream, "parens"); |
||||
|
||||
if (type == "hash" && !/^#([0-9a-fA-f]{3,4}|[0-9a-fA-f]{6}|[0-9a-fA-f]{8})$/.test(stream.current())) { |
||||
override += " error"; |
||||
} else if (type == "word") { |
||||
wordAsValue(stream); |
||||
} else if (type == "interpolation") { |
||||
return pushContext(state, stream, "interpolation"); |
||||
} |
||||
return "prop"; |
||||
}; |
||||
|
||||
states.propBlock = function(type, _stream, state) { |
||||
if (type == "}") return popContext(state); |
||||
if (type == "word") { override = "property"; return "maybeprop"; } |
||||
return state.context.type; |
||||
}; |
||||
|
||||
states.parens = function(type, stream, state) { |
||||
if (type == "{" || type == "}") return popAndPass(type, stream, state); |
||||
if (type == ")") return popContext(state); |
||||
if (type == "(") return pushContext(state, stream, "parens"); |
||||
if (type == "interpolation") return pushContext(state, stream, "interpolation"); |
||||
if (type == "word") wordAsValue(stream); |
||||
return "parens"; |
||||
}; |
||||
|
||||
states.pseudo = function(type, stream, state) { |
||||
if (type == "word") { |
||||
override = "variable-3"; |
||||
return state.context.type; |
||||
} |
||||
return pass(type, stream, state); |
||||
}; |
||||
|
||||
states.documentTypes = function(type, stream, state) { |
||||
if (type == "word" && documentTypes.hasOwnProperty(stream.current())) { |
||||
override = "tag"; |
||||
return state.context.type; |
||||
} else { |
||||
return states.atBlock(type, stream, state); |
||||
} |
||||
}; |
||||
|
||||
states.atBlock = function(type, stream, state) { |
||||
if (type == "(") return pushContext(state, stream, "atBlock_parens"); |
||||
if (type == "}" || type == ";") return popAndPass(type, stream, state); |
||||
if (type == "{") return popContext(state) && pushContext(state, stream, allowNested ? "block" : "top"); |
||||
|
||||
if (type == "interpolation") return pushContext(state, stream, "interpolation"); |
||||
|
||||
if (type == "word") { |
||||
var word = stream.current().toLowerCase(); |
||||
if (word == "only" || word == "not" || word == "and" || word == "or") |
||||
override = "keyword"; |
||||
else if (mediaTypes.hasOwnProperty(word)) |
||||
override = "attribute"; |
||||
else if (mediaFeatures.hasOwnProperty(word)) |
||||
override = "property"; |
||||
else if (mediaValueKeywords.hasOwnProperty(word)) |
||||
override = "keyword"; |
||||
else if (propertyKeywords.hasOwnProperty(word)) |
||||
override = "property"; |
||||
else if (nonStandardPropertyKeywords.hasOwnProperty(word)) |
||||
override = "string-2"; |
||||
else if (valueKeywords.hasOwnProperty(word)) |
||||
override = "atom"; |
||||
else if (colorKeywords.hasOwnProperty(word)) |
||||
override = "keyword"; |
||||
else |
||||
override = "error"; |
||||
} |
||||
return state.context.type; |
||||
}; |
||||
|
||||
states.atComponentBlock = function(type, stream, state) { |
||||
if (type == "}") |
||||
return popAndPass(type, stream, state); |
||||
if (type == "{") |
||||
return popContext(state) && pushContext(state, stream, allowNested ? "block" : "top", false); |
||||
if (type == "word") |
||||
override = "error"; |
||||
return state.context.type; |
||||
}; |
||||
|
||||
states.atBlock_parens = function(type, stream, state) { |
||||
if (type == ")") return popContext(state); |
||||
if (type == "{" || type == "}") return popAndPass(type, stream, state, 2); |
||||
return states.atBlock(type, stream, state); |
||||
}; |
||||
|
||||
states.restricted_atBlock_before = function(type, stream, state) { |
||||
if (type == "{") |
||||
return pushContext(state, stream, "restricted_atBlock"); |
||||
if (type == "word" && state.stateArg == "@counter-style") { |
||||
override = "variable"; |
||||
return "restricted_atBlock_before"; |
||||
} |
||||
return pass(type, stream, state); |
||||
}; |
||||
|
||||
states.restricted_atBlock = function(type, stream, state) { |
||||
if (type == "}") { |
||||
state.stateArg = null; |
||||
return popContext(state); |
||||
} |
||||
if (type == "word") { |
||||
if ((state.stateArg == "@font-face" && !fontProperties.hasOwnProperty(stream.current().toLowerCase())) || |
||||
(state.stateArg == "@counter-style" && !counterDescriptors.hasOwnProperty(stream.current().toLowerCase()))) |
||||
override = "error"; |
||||
else |
||||
override = "property"; |
||||
return "maybeprop"; |
||||
} |
||||
return "restricted_atBlock"; |
||||
}; |
||||
|
||||
states.keyframes = function(type, stream, state) { |
||||
if (type == "word") { override = "variable"; return "keyframes"; } |
||||
if (type == "{") return pushContext(state, stream, "top"); |
||||
return pass(type, stream, state); |
||||
}; |
||||
|
||||
states.at = function(type, stream, state) { |
||||
if (type == ";") return popContext(state); |
||||
if (type == "{" || type == "}") return popAndPass(type, stream, state); |
||||
if (type == "word") override = "tag"; |
||||
else if (type == "hash") override = "builtin"; |
||||
return "at"; |
||||
}; |
||||
|
||||
states.interpolation = function(type, stream, state) { |
||||
if (type == "}") return popContext(state); |
||||
if (type == "{" || type == ";") return popAndPass(type, stream, state); |
||||
if (type == "word") override = "variable"; |
||||
else if (type != "variable" && type != "(" && type != ")") override = "error"; |
||||
return "interpolation"; |
||||
}; |
||||
|
||||
return { |
||||
startState: function(base) { |
||||
return {tokenize: null, |
||||
state: inline ? "block" : "top", |
||||
stateArg: null, |
||||
context: new Context(inline ? "block" : "top", base || 0, null)}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (!state.tokenize && stream.eatSpace()) return null; |
||||
var 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) { |
||||
var cx = state.context, ch = textAfter && textAfter.charAt(0); |
||||
var indent = cx.indent; |
||||
if (cx.type == "prop" && (ch == "}" || ch == ")")) cx = cx.prev; |
||||
if (cx.prev) { |
||||
if (ch == "}" && (cx.type == "block" || cx.type == "top" || |
||||
cx.type == "interpolation" || cx.type == "restricted_atBlock")) { |
||||
// Resume indentation from parent context.
|
||||
cx = cx.prev; |
||||
indent = cx.indent; |
||||
} else if (ch == ")" && (cx.type == "parens" || cx.type == "atBlock_parens") || |
||||
ch == "{" && (cx.type == "at" || cx.type == "atBlock")) { |
||||
// Dedent relative to current context.
|
||||
indent = Math.max(0, cx.indent - indentUnit); |
||||
cx = cx.prev; |
||||
} |
||||
} |
||||
return indent; |
||||
}, |
||||
|
||||
electricChars: "}", |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/", |
||||
fold: "brace" |
||||
}; |
||||
}); |
||||
|
||||
function keySet(array) { |
||||
var keys = {}; |
||||
for (var i = 0; i < array.length; ++i) { |
||||
keys[array[i]] = true; |
||||
} |
||||
return keys; |
||||
} |
||||
|
||||
var documentTypes_ = [ |
||||
"domain", "regexp", "url", "url-prefix" |
||||
], documentTypes = keySet(documentTypes_); |
||||
|
||||
var mediaTypes_ = [ |
||||
"all", "aural", "braille", "handheld", "print", "projection", "screen", |
||||
"tty", "tv", "embossed" |
||||
], mediaTypes = keySet(mediaTypes_); |
||||
|
||||
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", "orientation", |
||||
"device-pixel-ratio", "min-device-pixel-ratio", "max-device-pixel-ratio", |
||||
"pointer", "any-pointer", "hover", "any-hover" |
||||
], mediaFeatures = keySet(mediaFeatures_); |
||||
|
||||
var mediaValueKeywords_ = [ |
||||
"landscape", "portrait", "none", "coarse", "fine", "on-demand", "hover", |
||||
"interlace", "progressive" |
||||
], mediaValueKeywords = keySet(mediaValueKeywords_); |
||||
|
||||
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-blend-mode", "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-rows", "grid-column", "grid-column-end", "grid-column-gap", |
||||
"grid-column-start", "grid-gap", "grid-row", "grid-row-end", "grid-row-gap", |
||||
"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-threshold", "shape-inside", "shape-margin", |
||||
"shape-outside", "size", "speak", "speak-as", "speak-header", |
||||
"speak-numeral", "speak-punctuation", "speech-rate", "stress", "string-set", |
||||
"tab-size", "table-layout", "target", "target-name", "target-new", |
||||
"target-position", "text-align", "text-align-last", "text-decoration", |
||||
"text-decoration-color", "text-decoration-line", "text-decoration-skip", |
||||
"text-decoration-style", "text-emphasis", "text-emphasis-color", |
||||
"text-emphasis-position", "text-emphasis-style", "text-height", |
||||
"text-indent", "text-justify", "text-outline", "text-overflow", "text-shadow", |
||||
"text-size-adjust", "text-space-collapse", "text-transform", "text-underline-position", |
||||
"text-wrap", "top", "transform", "transform-origin", "transform-style", |
||||
"transition", "transition-delay", "transition-duration", |
||||
"transition-property", "transition-timing-function", "unicode-bidi", |
||||
"vertical-align", "visibility", "voice-balance", "voice-duration", |
||||
"voice-family", "voice-pitch", "voice-range", "voice-rate", "voice-stress", |
||||
"voice-volume", "volume", "white-space", "widows", "width", "word-break", |
||||
"word-spacing", "word-wrap", "z-index", |
||||
// SVG-specific
|
||||
"clip-path", "clip-rule", "mask", "enable-background", "filter", "flood-color", |
||||
"flood-opacity", "lighting-color", "stop-color", "stop-opacity", "pointer-events", |
||||
"color-interpolation", "color-interpolation-filters", |
||||
"color-rendering", "fill", "fill-opacity", "fill-rule", "image-rendering", |
||||
"marker", "marker-end", "marker-mid", "marker-start", "shape-rendering", "stroke", |
||||
"stroke-dasharray", "stroke-dashoffset", "stroke-linecap", "stroke-linejoin", |
||||
"stroke-miterlimit", "stroke-opacity", "stroke-width", "text-rendering", |
||||
"baseline-shift", "dominant-baseline", "glyph-orientation-horizontal", |
||||
"glyph-orientation-vertical", "text-anchor", "writing-mode" |
||||
], propertyKeywords = keySet(propertyKeywords_); |
||||
|
||||
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" |
||||
], nonStandardPropertyKeywords = keySet(nonStandardPropertyKeywords_); |
||||
|
||||
var fontProperties_ = [ |
||||
"font-family", "src", "unicode-range", "font-variant", "font-feature-settings", |
||||
"font-stretch", "font-weight", "font-style" |
||||
], fontProperties = keySet(fontProperties_); |
||||
|
||||
var counterDescriptors_ = [ |
||||
"additive-symbols", "fallback", "negative", "pad", "prefix", "range", |
||||
"speak-as", "suffix", "symbols", "system" |
||||
], counterDescriptors = keySet(counterDescriptors_); |
||||
|
||||
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" |
||||
], colorKeywords = keySet(colorKeywords_); |
||||
|
||||
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", "color", "color-burn", "color-dodge", "column", "column-reverse", |
||||
"compact", "condensed", "contain", "content", |
||||
"content-box", "context-menu", "continuous", "copy", "counter", "counters", "cover", "crop", |
||||
"cross", "crosshair", "currentcolor", "cursive", "cyclic", "darken", "dashed", "decimal", |
||||
"decimal-leading-zero", "default", "default-button", "dense", "destination-atop", |
||||
"destination-in", "destination-out", "destination-over", "devanagari", "difference", |
||||
"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", "exclusion", "expanded", "extends", "extra-condensed", |
||||
"extra-expanded", "fantasy", "fast", "fill", "fixed", "flat", "flex", "flex-end", "flex-start", "footnotes", |
||||
"forwards", "from", "geometricPrecision", "georgian", "graytext", "grid", "groove", |
||||
"gujarati", "gurmukhi", "hand", "hangul", "hangul-consonant", "hard-light", "hebrew", |
||||
"help", "hidden", "hide", "higher", "highlight", "highlighttext", |
||||
"hiragana", "hiragana-iroha", "horizontal", "hsl", "hsla", "hue", "icon", "ignore", |
||||
"inactiveborder", "inactivecaption", "inactivecaptiontext", "infinite", |
||||
"infobackground", "infotext", "inherit", "initial", "inline", "inline-axis", |
||||
"inline-block", "inline-flex", "inline-grid", "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", "lighten", |
||||
"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", "luminosity", "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", "multiply", "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-button", "radial-gradient", "radio", "read-only", |
||||
"read-write", "read-write-plaintext-only", "rectangle", "region", |
||||
"relative", "repeat", "repeating-linear-gradient", |
||||
"repeating-radial-gradient", "repeat-x", "repeat-y", "reset", "reverse", |
||||
"rgb", "rgba", "ridge", "right", "rotate", "rotate3d", "rotateX", "rotateY", |
||||
"rotateZ", "round", "row", "row-resize", "row-reverse", "rtl", "run-in", "running", |
||||
"s-resize", "sans-serif", "saturation", "scale", "scale3d", "scaleX", "scaleY", "scaleZ", "screen", |
||||
"scroll", "scrollbar", "se-resize", "searchfield", |
||||
"searchfield-cancel-button", "searchfield-decoration", |
||||
"searchfield-results-button", "searchfield-results-decoration", |
||||
"semi-condensed", "semi-expanded", "separate", "serif", "show", "sidama", |
||||
"simp-chinese-formal", "simp-chinese-informal", "single", |
||||
"skew", "skewX", "skewY", "skip-white-space", "slide", "slider-horizontal", |
||||
"slider-vertical", "sliderthumb-horizontal", "sliderthumb-vertical", "slow", |
||||
"small", "small-caps", "small-caption", "smaller", "soft-light", "solid", "somali", |
||||
"source-atop", "source-in", "source-out", "source-over", "space", "space-around", "space-between", "spell-out", "square", |
||||
"square-button", "start", "static", "status-bar", "stretch", "stroke", "sub", |
||||
"subpixel-antialiased", "super", "sw-resize", "symbolic", "symbols", "table", |
||||
"table-caption", "table-cell", "table-column", "table-column-group", |
||||
"table-footer-group", "table-header-group", "table-row", "table-row-group", |
||||
"tamil", |
||||
"telugu", "text", "text-bottom", "text-top", "textarea", "textfield", "thai", |
||||
"thick", "thin", "threeddarkshadow", "threedface", "threedhighlight", |
||||
"threedlightshadow", "threedshadow", "tibetan", "tigre", "tigrinya-er", |
||||
"tigrinya-er-abegede", "tigrinya-et", "tigrinya-et-abegede", "to", "top", |
||||
"trad-chinese-formal", "trad-chinese-informal", |
||||
"translate", "translate3d", "translateX", "translateY", "translateZ", |
||||
"transparent", "ultra-condensed", "ultra-expanded", "underline", "up", |
||||
"upper-alpha", "upper-armenian", "upper-greek", "upper-hexadecimal", |
||||
"upper-latin", "upper-norwegian", "upper-roman", "uppercase", "urdu", "url", |
||||
"var", "vertical", "vertical-text", "visible", "visibleFill", "visiblePainted", |
||||
"visibleStroke", "visual", "w-resize", "wait", "wave", "wider", |
||||
"window", "windowframe", "windowtext", "words", "wrap", "wrap-reverse", "x-large", "x-small", "xor", |
||||
"xx-large", "xx-small" |
||||
], valueKeywords = keySet(valueKeywords_); |
||||
|
||||
var allWords = documentTypes_.concat(mediaTypes_).concat(mediaFeatures_).concat(mediaValueKeywords_) |
||||
.concat(propertyKeywords_).concat(nonStandardPropertyKeywords_).concat(colorKeywords_) |
||||
.concat(valueKeywords_); |
||||
CodeMirror.registerHelper("hintWords", "css", allWords); |
||||
|
||||
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"]; |
||||
} |
||||
|
||||
CodeMirror.defineMIME("text/css", { |
||||
documentTypes: documentTypes, |
||||
mediaTypes: mediaTypes, |
||||
mediaFeatures: mediaFeatures, |
||||
mediaValueKeywords: mediaValueKeywords, |
||||
propertyKeywords: propertyKeywords, |
||||
nonStandardPropertyKeywords: nonStandardPropertyKeywords, |
||||
fontProperties: fontProperties, |
||||
counterDescriptors: counterDescriptors, |
||||
colorKeywords: colorKeywords, |
||||
valueKeywords: valueKeywords, |
||||
tokenHooks: { |
||||
"/": function(stream, state) { |
||||
if (!stream.eat("*")) return false; |
||||
state.tokenize = tokenCComment; |
||||
return tokenCComment(stream, state); |
||||
} |
||||
}, |
||||
name: "css" |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-scss", { |
||||
mediaTypes: mediaTypes, |
||||
mediaFeatures: mediaFeatures, |
||||
mediaValueKeywords: mediaValueKeywords, |
||||
propertyKeywords: propertyKeywords, |
||||
nonStandardPropertyKeywords: nonStandardPropertyKeywords, |
||||
colorKeywords: colorKeywords, |
||||
valueKeywords: valueKeywords, |
||||
fontProperties: fontProperties, |
||||
allowNested: true, |
||||
tokenHooks: { |
||||
"/": function(stream, state) { |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return ["comment", "comment"]; |
||||
} else if (stream.eat("*")) { |
||||
state.tokenize = tokenCComment; |
||||
return tokenCComment(stream, state); |
||||
} else { |
||||
return ["operator", "operator"]; |
||||
} |
||||
}, |
||||
":": function(stream) { |
||||
if (stream.match(/\s*\{/)) |
||||
return [null, "{"]; |
||||
return false; |
||||
}, |
||||
"$": function(stream) { |
||||
stream.match(/^[\w-]+/); |
||||
if (stream.match(/^\s*:/, false)) |
||||
return ["variable-2", "variable-definition"]; |
||||
return ["variable-2", "variable"]; |
||||
}, |
||||
"#": function(stream) { |
||||
if (!stream.eat("{")) return false; |
||||
return [null, "interpolation"]; |
||||
} |
||||
}, |
||||
name: "css", |
||||
helperType: "scss" |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-less", { |
||||
mediaTypes: mediaTypes, |
||||
mediaFeatures: mediaFeatures, |
||||
mediaValueKeywords: mediaValueKeywords, |
||||
propertyKeywords: propertyKeywords, |
||||
nonStandardPropertyKeywords: nonStandardPropertyKeywords, |
||||
colorKeywords: colorKeywords, |
||||
valueKeywords: valueKeywords, |
||||
fontProperties: fontProperties, |
||||
allowNested: true, |
||||
tokenHooks: { |
||||
"/": function(stream, state) { |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return ["comment", "comment"]; |
||||
} else if (stream.eat("*")) { |
||||
state.tokenize = tokenCComment; |
||||
return tokenCComment(stream, state); |
||||
} else { |
||||
return ["operator", "operator"]; |
||||
} |
||||
}, |
||||
"@": function(stream) { |
||||
if (stream.eat("{")) return [null, "interpolation"]; |
||||
if (stream.match(/^(charset|document|font-face|import|(-(moz|ms|o|webkit)-)?keyframes|media|namespace|page|supports)\b/, false)) return false; |
||||
stream.eatWhile(/[\w\\\-]/); |
||||
if (stream.match(/^\s*:/, false)) |
||||
return ["variable-2", "variable-definition"]; |
||||
return ["variable-2", "variable"]; |
||||
}, |
||||
"&": function() { |
||||
return ["atom", "atom"]; |
||||
} |
||||
}, |
||||
name: "css", |
||||
helperType: "less" |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-gss", { |
||||
documentTypes: documentTypes, |
||||
mediaTypes: mediaTypes, |
||||
mediaFeatures: mediaFeatures, |
||||
propertyKeywords: propertyKeywords, |
||||
nonStandardPropertyKeywords: nonStandardPropertyKeywords, |
||||
fontProperties: fontProperties, |
||||
counterDescriptors: counterDescriptors, |
||||
colorKeywords: colorKeywords, |
||||
valueKeywords: valueKeywords, |
||||
supportsAtComponent: true, |
||||
tokenHooks: { |
||||
"/": function(stream, state) { |
||||
if (!stream.eat("*")) return false; |
||||
state.tokenize = tokenCComment; |
||||
return tokenCComment(stream, state); |
||||
} |
||||
}, |
||||
name: "css", |
||||
helperType: "gss" |
||||
}); |
||||
|
||||
}); |
@ -0,0 +1,103 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Closure Stylesheets (GSS) mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../addon/hint/show-hint.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="css.js"></script> |
||||
<script src="../../addon/hint/show-hint.js"></script> |
||||
<script src="../../addon/hint/css-hint.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Closure Stylesheets (GSS)</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Closure Stylesheets (GSS) mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* Some example Closure Stylesheets */ |
||||
|
||||
@provide 'some.styles'; |
||||
|
||||
@require 'other.styles'; |
||||
|
||||
@component { |
||||
|
||||
@def FONT_FAMILY "Times New Roman", Georgia, Serif; |
||||
@def FONT_SIZE_NORMAL 15px; |
||||
@def FONT_NORMAL normal FONT_SIZE_NORMAL FONT_FAMILY; |
||||
|
||||
@def BG_COLOR rgb(235, 239, 249); |
||||
|
||||
@def DIALOG_BORDER_COLOR rgb(107, 144, 218); |
||||
@def DIALOG_BG_COLOR BG_COLOR; |
||||
|
||||
@def LEFT_HAND_NAV_WIDTH 180px; |
||||
@def LEFT_HAND_NAV_PADDING 3px; |
||||
|
||||
@defmixin size(WIDTH, HEIGHT) { |
||||
width: WIDTH; |
||||
height: HEIGHT; |
||||
} |
||||
|
||||
body { |
||||
background-color: BG_COLOR; |
||||
margin: 0; |
||||
padding: 3em 6em; |
||||
font: FONT_NORMAL; |
||||
color: #000; |
||||
} |
||||
|
||||
#navigation a { |
||||
font-weight: bold; |
||||
text-decoration: none !important; |
||||
} |
||||
|
||||
.dialog { |
||||
background-color: DIALOG_BG_COLOR; |
||||
border: 1px solid DIALOG_BORDER_COLOR; |
||||
} |
||||
|
||||
.content { |
||||
position: absolute; |
||||
margin-left: add(LEFT_HAND_NAV_PADDING, /* padding left */ |
||||
LEFT_HAND_NAV_WIDTH, |
||||
LEFT_HAND_NAV_PADDING); /* padding right */ |
||||
|
||||
} |
||||
|
||||
.logo { |
||||
@mixin size(150px, 55px); |
||||
background-image: url('http://www.google.com/images/logo_sm.gif'); |
||||
} |
||||
|
||||
} |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
extraKeys: {"Ctrl-Space": "autocomplete"}, |
||||
lineNumbers: true, |
||||
matchBrackets: "text/x-less", |
||||
mode: "text/x-gss" |
||||
}); |
||||
</script> |
||||
|
||||
<p>A mode for <a href="https://github.com/google/closure-stylesheets">Closure Stylesheets</a> (GSS).</p> |
||||
<p><strong>MIME type defined:</strong> <code>text/x-gss</code>.</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#gss_*">normal</a>, <a href="../../test/index.html#verbose,gss_*">verbose</a>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,17 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
"use strict"; |
||||
|
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "text/x-gss"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1), "gss"); } |
||||
|
||||
MT("atComponent", |
||||
"[def @component] {", |
||||
"[tag foo] {", |
||||
" [property color]: [keyword black];", |
||||
"}", |
||||
"}"); |
||||
|
||||
})(); |
@ -0,0 +1,75 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: CSS mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../addon/hint/show-hint.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="css.js"></script> |
||||
<script src="../../addon/hint/show-hint.js"></script> |
||||
<script src="../../addon/hint/css-hint.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">CSS</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>CSS mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* Some example CSS */ |
||||
|
||||
@import url("something.css"); |
||||
|
||||
body { |
||||
margin: 0; |
||||
padding: 3em 6em; |
||||
font-family: tahoma, arial, sans-serif; |
||||
color: #000; |
||||
} |
||||
|
||||
#navigation a { |
||||
font-weight: bold; |
||||
text-decoration: none !important; |
||||
} |
||||
|
||||
h1 { |
||||
font-size: 2.5em; |
||||
} |
||||
|
||||
h2 { |
||||
font-size: 1.7em; |
||||
} |
||||
|
||||
h1:before, h2:before { |
||||
content: "::"; |
||||
} |
||||
|
||||
code { |
||||
font-family: courier, monospace; |
||||
font-size: 80%; |
||||
color: #418A8A; |
||||
} |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
extraKeys: {"Ctrl-Space": "autocomplete"}, |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/css</code>, <code>text/x-scss</code> (<a href="scss.html">demo</a>), <code>text/x-less</code> (<a href="less.html">demo</a>).</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#css_*">normal</a>, <a href="../../test/index.html#verbose,css_*">verbose</a>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,152 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: LESS mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="css.js"></script> |
||||
<style>.CodeMirror {border: 1px solid #ddd; line-height: 1.2;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">LESS</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>LESS mode</h2> |
||||
<form><textarea id="code" name="code">@media screen and (device-aspect-ratio: 16/9) { … } |
||||
@media screen and (device-aspect-ratio: 1280/720) { … } |
||||
@media screen and (device-aspect-ratio: 2560/1440) { … } |
||||
|
||||
html:lang(fr-be) |
||||
|
||||
tr:nth-child(2n+1) /* represents every odd row of an HTML table */ |
||||
|
||||
img:nth-of-type(2n+1) { float: right; } |
||||
img:nth-of-type(2n) { float: left; } |
||||
|
||||
body > h2:not(:first-of-type):not(:last-of-type) |
||||
|
||||
html|*:not(:link):not(:visited) |
||||
*|*:not(:hover) |
||||
p::first-line { text-transform: uppercase } |
||||
|
||||
@namespace foo url(http://www.example.com); |
||||
foo|h1 { color: blue } /* first rule */ |
||||
|
||||
span[hello="Ocean"][goodbye="Land"] |
||||
|
||||
E[foo]{ |
||||
padding:65px; |
||||
} |
||||
|
||||
input[type="search"]::-webkit-search-decoration, |
||||
input[type="search"]::-webkit-search-cancel-button { |
||||
-webkit-appearance: none; // Inner-padding issues in Chrome OSX, Safari 5 |
||||
} |
||||
button::-moz-focus-inner, |
||||
input::-moz-focus-inner { // Inner padding and border oddities in FF3/4 |
||||
padding: 0; |
||||
border: 0; |
||||
} |
||||
.btn { |
||||
// reset here as of 2.0.3 due to Recess property order |
||||
border-color: #ccc; |
||||
border-color: rgba(0,0,0,.1) rgba(0,0,0,.1) rgba(0,0,0,.25); |
||||
} |
||||
fieldset span button, fieldset span input[type="file"] { |
||||
font-size:12px; |
||||
font-family:Arial, Helvetica, sans-serif; |
||||
} |
||||
|
||||
.rounded-corners (@radius: 5px) { |
||||
border-radius: @radius; |
||||
-webkit-border-radius: @radius; |
||||
-moz-border-radius: @radius; |
||||
} |
||||
|
||||
@import url("something.css"); |
||||
|
||||
@light-blue: hsl(190, 50%, 65%); |
||||
|
||||
#menu { |
||||
position: absolute; |
||||
width: 100%; |
||||
z-index: 3; |
||||
clear: both; |
||||
display: block; |
||||
background-color: @blue; |
||||
height: 42px; |
||||
border-top: 2px solid lighten(@alpha-blue, 20%); |
||||
border-bottom: 2px solid darken(@alpha-blue, 25%); |
||||
.box-shadow(0, 1px, 8px, 0.6); |
||||
-moz-box-shadow: 0 0 0 #000; // Because firefox sucks. |
||||
|
||||
&.docked { |
||||
background-color: hsla(210, 60%, 40%, 0.4); |
||||
} |
||||
&:hover { |
||||
background-color: @blue; |
||||
} |
||||
|
||||
#dropdown { |
||||
margin: 0 0 0 117px; |
||||
padding: 0; |
||||
padding-top: 5px; |
||||
display: none; |
||||
width: 190px; |
||||
border-top: 2px solid @medium; |
||||
color: @highlight; |
||||
border: 2px solid darken(@medium, 25%); |
||||
border-left-color: darken(@medium, 15%); |
||||
border-right-color: darken(@medium, 15%); |
||||
border-top-width: 0; |
||||
background-color: darken(@medium, 10%); |
||||
ul { |
||||
padding: 0px; |
||||
} |
||||
li { |
||||
font-size: 14px; |
||||
display: block; |
||||
text-align: left; |
||||
padding: 0; |
||||
border: 0; |
||||
a { |
||||
display: block; |
||||
padding: 0px 15px; |
||||
text-decoration: none; |
||||
color: white; |
||||
&:hover { |
||||
background-color: darken(@medium, 15%); |
||||
text-decoration: none; |
||||
} |
||||
} |
||||
} |
||||
.border-radius(5px, bottom); |
||||
.box-shadow(0, 6px, 8px, 0.5); |
||||
} |
||||
} |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers : true, |
||||
matchBrackets : true, |
||||
mode: "text/x-less" |
||||
}); |
||||
</script> |
||||
|
||||
<p>The LESS mode is a sub-mode of the <a href="index.html">CSS mode</a> (defined in <code>css.js</code>).</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#less_*">normal</a>, <a href="../../test/index.html#verbose,less_*">verbose</a>.</p> |
||||
</article> |
@ -0,0 +1,54 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
"use strict"; |
||||
|
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "text/x-less"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1), "less"); } |
||||
|
||||
MT("variable", |
||||
"[variable-2 @base]: [atom #f04615];", |
||||
"[qualifier .class] {", |
||||
" [property width]: [variable percentage]([number 0.5]); [comment // returns `50%`]", |
||||
" [property color]: [variable saturate]([variable-2 @base], [number 5%]);", |
||||
"}"); |
||||
|
||||
MT("amp", |
||||
"[qualifier .child], [qualifier .sibling] {", |
||||
" [qualifier .parent] [atom &] {", |
||||
" [property color]: [keyword black];", |
||||
" }", |
||||
" [atom &] + [atom &] {", |
||||
" [property color]: [keyword red];", |
||||
" }", |
||||
"}"); |
||||
|
||||
MT("mixin", |
||||
"[qualifier .mixin] ([variable dark]; [variable-2 @color]) {", |
||||
" [property color]: [atom darken]([variable-2 @color], [number 10%]);", |
||||
"}", |
||||
"[qualifier .mixin] ([variable light]; [variable-2 @color]) {", |
||||
" [property color]: [atom lighten]([variable-2 @color], [number 10%]);", |
||||
"}", |
||||
"[qualifier .mixin] ([variable-2 @_]; [variable-2 @color]) {", |
||||
" [property display]: [atom block];", |
||||
"}", |
||||
"[variable-2 @switch]: [variable light];", |
||||
"[qualifier .class] {", |
||||
" [qualifier .mixin]([variable-2 @switch]; [atom #888]);", |
||||
"}"); |
||||
|
||||
MT("nest", |
||||
"[qualifier .one] {", |
||||
" [def @media] ([property width]: [number 400px]) {", |
||||
" [property font-size]: [number 1.2em];", |
||||
" [def @media] [attribute print] [keyword and] [property color] {", |
||||
" [property color]: [keyword blue];", |
||||
" }", |
||||
" }", |
||||
"}"); |
||||
|
||||
|
||||
MT("interpolation", ".@{[variable foo]} { [property font-weight]: [atom bold]; }"); |
||||
})(); |
@ -0,0 +1,157 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: SCSS mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="css.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">SCSS</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>SCSS mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* Some example SCSS */ |
||||
|
||||
@import "compass/css3"; |
||||
$variable: #333; |
||||
|
||||
$blue: #3bbfce; |
||||
$margin: 16px; |
||||
|
||||
.content-navigation { |
||||
#nested { |
||||
background-color: black; |
||||
} |
||||
border-color: $blue; |
||||
color: |
||||
darken($blue, 9%); |
||||
} |
||||
|
||||
.border { |
||||
padding: $margin / 2; |
||||
margin: $margin / 2; |
||||
border-color: $blue; |
||||
} |
||||
|
||||
@mixin table-base { |
||||
th { |
||||
text-align: center; |
||||
font-weight: bold; |
||||
} |
||||
td, th {padding: 2px} |
||||
} |
||||
|
||||
table.hl { |
||||
margin: 2em 0; |
||||
td.ln { |
||||
text-align: right; |
||||
} |
||||
} |
||||
|
||||
li { |
||||
font: { |
||||
family: serif; |
||||
weight: bold; |
||||
size: 1.2em; |
||||
} |
||||
} |
||||
|
||||
@mixin left($dist) { |
||||
float: left; |
||||
margin-left: $dist; |
||||
} |
||||
|
||||
#data { |
||||
@include left(10px); |
||||
@include table-base; |
||||
} |
||||
|
||||
.source { |
||||
@include flow-into(target); |
||||
border: 10px solid green; |
||||
margin: 20px; |
||||
width: 200px; } |
||||
|
||||
.new-container { |
||||
@include flow-from(target); |
||||
border: 10px solid red; |
||||
margin: 20px; |
||||
width: 200px; } |
||||
|
||||
body { |
||||
margin: 0; |
||||
padding: 3em 6em; |
||||
font-family: tahoma, arial, sans-serif; |
||||
color: #000; |
||||
} |
||||
|
||||
@mixin yellow() { |
||||
background: yellow; |
||||
} |
||||
|
||||
.big { |
||||
font-size: 14px; |
||||
} |
||||
|
||||
.nested { |
||||
@include border-radius(3px); |
||||
@extend .big; |
||||
p { |
||||
background: whitesmoke; |
||||
a { |
||||
color: red; |
||||
} |
||||
} |
||||
} |
||||
|
||||
#navigation a { |
||||
font-weight: bold; |
||||
text-decoration: none !important; |
||||
} |
||||
|
||||
h1 { |
||||
font-size: 2.5em; |
||||
} |
||||
|
||||
h2 { |
||||
font-size: 1.7em; |
||||
} |
||||
|
||||
h1:before, h2:before { |
||||
content: "::"; |
||||
} |
||||
|
||||
code { |
||||
font-family: courier, monospace; |
||||
font-size: 80%; |
||||
color: #418A8A; |
||||
} |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-scss" |
||||
}); |
||||
</script> |
||||
|
||||
<p>The SCSS mode is a sub-mode of the <a href="index.html">CSS mode</a> (defined in <code>css.js</code>).</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#scss_*">normal</a>, <a href="../../test/index.html#verbose,scss_*">verbose</a>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,110 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "text/x-scss"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1), "scss"); } |
||||
|
||||
MT('url_with_quotation', |
||||
"[tag foo] { [property background]:[atom url]([string test.jpg]) }"); |
||||
|
||||
MT('url_with_double_quotes', |
||||
"[tag foo] { [property background]:[atom url]([string \"test.jpg\"]) }"); |
||||
|
||||
MT('url_with_single_quotes', |
||||
"[tag foo] { [property background]:[atom url]([string \'test.jpg\']) }"); |
||||
|
||||
MT('string', |
||||
"[def @import] [string \"compass/css3\"]"); |
||||
|
||||
MT('important_keyword', |
||||
"[tag foo] { [property background]:[atom url]([string \'test.jpg\']) [keyword !important] }"); |
||||
|
||||
MT('variable', |
||||
"[variable-2 $blue]:[atom #333]"); |
||||
|
||||
MT('variable_as_attribute', |
||||
"[tag foo] { [property color]:[variable-2 $blue] }"); |
||||
|
||||
MT('numbers', |
||||
"[tag foo] { [property padding]:[number 10px] [number 10] [number 10em] [number 8in] }"); |
||||
|
||||
MT('number_percentage', |
||||
"[tag foo] { [property width]:[number 80%] }"); |
||||
|
||||
MT('selector', |
||||
"[builtin #hello][qualifier .world]{}"); |
||||
|
||||
MT('singleline_comment', |
||||
"[comment // this is a comment]"); |
||||
|
||||
MT('multiline_comment', |
||||
"[comment /*foobar*/]"); |
||||
|
||||
MT('attribute_with_hyphen', |
||||
"[tag foo] { [property font-size]:[number 10px] }"); |
||||
|
||||
MT('string_after_attribute', |
||||
"[tag foo] { [property content]:[string \"::\"] }"); |
||||
|
||||
MT('directives', |
||||
"[def @include] [qualifier .mixin]"); |
||||
|
||||
MT('basic_structure', |
||||
"[tag p] { [property background]:[keyword red]; }"); |
||||
|
||||
MT('nested_structure', |
||||
"[tag p] { [tag a] { [property color]:[keyword red]; } }"); |
||||
|
||||
MT('mixin', |
||||
"[def @mixin] [tag table-base] {}"); |
||||
|
||||
MT('number_without_semicolon', |
||||
"[tag p] {[property width]:[number 12]}", |
||||
"[tag a] {[property color]:[keyword red];}"); |
||||
|
||||
MT('atom_in_nested_block', |
||||
"[tag p] { [tag a] { [property color]:[atom #000]; } }"); |
||||
|
||||
MT('interpolation_in_property', |
||||
"[tag foo] { #{[variable-2 $hello]}:[number 2]; }"); |
||||
|
||||
MT('interpolation_in_selector', |
||||
"[tag foo]#{[variable-2 $hello]} { [property color]:[atom #000]; }"); |
||||
|
||||
MT('interpolation_error', |
||||
"[tag foo]#{[variable foo]} { [property color]:[atom #000]; }"); |
||||
|
||||
MT("divide_operator", |
||||
"[tag foo] { [property width]:[number 4] [operator /] [number 2] }"); |
||||
|
||||
MT('nested_structure_with_id_selector', |
||||
"[tag p] { [builtin #hello] { [property color]:[keyword red]; } }"); |
||||
|
||||
MT('indent_mixin', |
||||
"[def @mixin] [tag container] (", |
||||
" [variable-2 $a]: [number 10],", |
||||
" [variable-2 $b]: [number 10])", |
||||
"{}"); |
||||
|
||||
MT('indent_nested', |
||||
"[tag foo] {", |
||||
" [tag bar] {", |
||||
" }", |
||||
"}"); |
||||
|
||||
MT('indent_parentheses', |
||||
"[tag foo] {", |
||||
" [property color]: [atom darken]([variable-2 $blue],", |
||||
" [number 9%]);", |
||||
"}"); |
||||
|
||||
MT('indent_vardef', |
||||
"[variable-2 $name]:", |
||||
" [string 'val'];", |
||||
"[tag tag] {", |
||||
" [tag inner] {", |
||||
" [property margin]: [number 3px];", |
||||
" }", |
||||
"}"); |
||||
})(); |
@ -0,0 +1,200 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "css"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
// Error, because "foobarhello" is neither a known type or property, but
|
||||
// property was expected (after "and"), and it should be in parentheses.
|
||||
MT("atMediaUnknownType", |
||||
"[def @media] [attribute screen] [keyword and] [error foobarhello] { }"); |
||||
|
||||
// Soft error, because "foobarhello" is not a known property or type.
|
||||
MT("atMediaUnknownProperty", |
||||
"[def @media] [attribute screen] [keyword and] ([error foobarhello]) { }"); |
||||
|
||||
// Make sure nesting works with media queries
|
||||
MT("atMediaMaxWidthNested", |
||||
"[def @media] [attribute screen] [keyword and] ([property max-width]: [number 25px]) { [tag foo] { } }"); |
||||
|
||||
MT("atMediaFeatureValueKeyword", |
||||
"[def @media] ([property orientation]: [keyword landscape]) { }"); |
||||
|
||||
MT("atMediaUnknownFeatureValueKeyword", |
||||
"[def @media] ([property orientation]: [error upsidedown]) { }"); |
||||
|
||||
MT("tagSelector", |
||||
"[tag foo] { }"); |
||||
|
||||
MT("classSelector", |
||||
"[qualifier .foo-bar_hello] { }"); |
||||
|
||||
MT("idSelector", |
||||
"[builtin #foo] { [error #foo] }"); |
||||
|
||||
MT("tagSelectorUnclosed", |
||||
"[tag foo] { [property margin]: [number 0] } [tag bar] { }"); |
||||
|
||||
MT("tagStringNoQuotes", |
||||
"[tag foo] { [property font-family]: [variable hello] [variable world]; }"); |
||||
|
||||
MT("tagStringDouble", |
||||
"[tag foo] { [property font-family]: [string \"hello world\"]; }"); |
||||
|
||||
MT("tagStringSingle", |
||||
"[tag foo] { [property font-family]: [string 'hello world']; }"); |
||||
|
||||
MT("tagColorKeyword", |
||||
"[tag foo] {", |
||||
" [property color]: [keyword black];", |
||||
" [property color]: [keyword navy];", |
||||
" [property color]: [keyword yellow];", |
||||
"}"); |
||||
|
||||
MT("tagColorHex3", |
||||
"[tag foo] { [property background]: [atom #fff]; }"); |
||||
|
||||
MT("tagColorHex4", |
||||
"[tag foo] { [property background]: [atom #ffff]; }"); |
||||
|
||||
MT("tagColorHex6", |
||||
"[tag foo] { [property background]: [atom #ffffff]; }"); |
||||
|
||||
MT("tagColorHex8", |
||||
"[tag foo] { [property background]: [atom #ffffffff]; }"); |
||||
|
||||
MT("tagColorHex5Invalid", |
||||
"[tag foo] { [property background]: [atom&error #fffff]; }"); |
||||
|
||||
MT("tagColorHexInvalid", |
||||
"[tag foo] { [property background]: [atom&error #ffg]; }"); |
||||
|
||||
MT("tagNegativeNumber", |
||||
"[tag foo] { [property margin]: [number -5px]; }"); |
||||
|
||||
MT("tagPositiveNumber", |
||||
"[tag foo] { [property padding]: [number 5px]; }"); |
||||
|
||||
MT("tagVendor", |
||||
"[tag foo] { [meta -foo-][property box-sizing]: [meta -foo-][atom border-box]; }"); |
||||
|
||||
MT("tagBogusProperty", |
||||
"[tag foo] { [property&error barhelloworld]: [number 0]; }"); |
||||
|
||||
MT("tagTwoProperties", |
||||
"[tag foo] { [property margin]: [number 0]; [property padding]: [number 0]; }"); |
||||
|
||||
MT("tagTwoPropertiesURL", |
||||
"[tag foo] { [property background]: [atom url]([string //example.com/foo.png]); [property padding]: [number 0]; }"); |
||||
|
||||
MT("indent_tagSelector", |
||||
"[tag strong], [tag em] {", |
||||
" [property background]: [atom rgba](", |
||||
" [number 255], [number 255], [number 0], [number .2]", |
||||
" );", |
||||
"}"); |
||||
|
||||
MT("indent_atMedia", |
||||
"[def @media] {", |
||||
" [tag foo] {", |
||||
" [property color]:", |
||||
" [keyword yellow];", |
||||
" }", |
||||
"}"); |
||||
|
||||
MT("indent_comma", |
||||
"[tag foo] {", |
||||
" [property font-family]: [variable verdana],", |
||||
" [atom sans-serif];", |
||||
"}"); |
||||
|
||||
MT("indent_parentheses", |
||||
"[tag foo]:[variable-3 before] {", |
||||
" [property background]: [atom url](", |
||||
"[string blahblah]", |
||||
"[string etc]", |
||||
"[string ]) [keyword !important];", |
||||
"}"); |
||||
|
||||
MT("font_face", |
||||
"[def @font-face] {", |
||||
" [property font-family]: [string 'myfont'];", |
||||
" [error nonsense]: [string 'abc'];", |
||||
" [property src]: [atom url]([string http://blah]),", |
||||
" [atom url]([string http://foo]);", |
||||
"}"); |
||||
|
||||
MT("empty_url", |
||||
"[def @import] [atom url]() [attribute screen];"); |
||||
|
||||
MT("parens", |
||||
"[qualifier .foo] {", |
||||
" [property background-image]: [variable fade]([atom #000], [number 20%]);", |
||||
" [property border-image]: [atom linear-gradient](", |
||||
" [atom to] [atom bottom],", |
||||
" [variable fade]([atom #000], [number 20%]) [number 0%],", |
||||
" [variable fade]([atom #000], [number 20%]) [number 100%]", |
||||
" );", |
||||
"}"); |
||||
|
||||
MT("css_variable", |
||||
":[variable-3 root] {", |
||||
" [variable-2 --main-color]: [atom #06c];", |
||||
"}", |
||||
"[tag h1][builtin #foo] {", |
||||
" [property color]: [atom var]([variable-2 --main-color]);", |
||||
"}"); |
||||
|
||||
MT("supports", |
||||
"[def @supports] ([keyword not] (([property text-align-last]: [atom justify]) [keyword or] ([meta -moz-][property text-align-last]: [atom justify])) {", |
||||
" [property text-align-last]: [atom justify];", |
||||
"}"); |
||||
|
||||
MT("document", |
||||
"[def @document] [tag url]([string http://blah]),", |
||||
" [tag url-prefix]([string https://]),", |
||||
" [tag domain]([string blah.com]),", |
||||
" [tag regexp]([string \".*blah.+\"]) {", |
||||
" [builtin #id] {", |
||||
" [property background-color]: [keyword white];", |
||||
" }", |
||||
" [tag foo] {", |
||||
" [property font-family]: [variable Verdana], [atom sans-serif];", |
||||
" }", |
||||
"}"); |
||||
|
||||
MT("document_url", |
||||
"[def @document] [tag url]([string http://blah]) { [qualifier .class] { } }"); |
||||
|
||||
MT("document_urlPrefix", |
||||
"[def @document] [tag url-prefix]([string https://]) { [builtin #id] { } }"); |
||||
|
||||
MT("document_domain", |
||||
"[def @document] [tag domain]([string blah.com]) { [tag foo] { } }"); |
||||
|
||||
MT("document_regexp", |
||||
"[def @document] [tag regexp]([string \".*blah.+\"]) { [builtin #id] { } }"); |
||||
|
||||
MT("counter-style", |
||||
"[def @counter-style] [variable binary] {", |
||||
" [property system]: [atom numeric];", |
||||
" [property symbols]: [number 0] [number 1];", |
||||
" [property suffix]: [string \".\"];", |
||||
" [property range]: [atom infinite];", |
||||
" [property speak-as]: [atom numeric];", |
||||
"}"); |
||||
|
||||
MT("counter-style-additive-symbols", |
||||
"[def @counter-style] [variable simple-roman] {", |
||||
" [property system]: [atom additive];", |
||||
" [property additive-symbols]: [number 10] [variable X], [number 5] [variable V], [number 1] [variable I];", |
||||
" [property range]: [number 1] [number 49];", |
||||
"}"); |
||||
|
||||
MT("counter-style-use", |
||||
"[tag ol][qualifier .roman] { [property list-style]: [variable simple-roman]; }"); |
||||
|
||||
MT("counter-style-symbols", |
||||
"[tag ol] { [property list-style]: [atom symbols]([atom cyclic] [string \"*\"] [string \"\\2020\"] [string \"\\2021\"] [string \"\\A7\"]); }"); |
||||
})(); |
@ -0,0 +1,146 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
// By the Neo4j Team and contributors.
|
||||
// https://github.com/neo4j-contrib/CodeMirror
|
||||
|
||||
(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"; |
||||
var wordRegexp = function(words) { |
||||
return new RegExp("^(?:" + words.join("|") + ")$", "i"); |
||||
}; |
||||
|
||||
CodeMirror.defineMode("cypher", function(config) { |
||||
var tokenBase = function(stream/*, state*/) { |
||||
var ch = stream.next(); |
||||
if (ch === "\"" || ch === "'") { |
||||
stream.match(/.+?["']/); |
||||
return "string"; |
||||
} |
||||
if (/[{}\(\),\.;\[\]]/.test(ch)) { |
||||
curPunc = ch; |
||||
return "node"; |
||||
} else if (ch === "/" && stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} else if (operatorChars.test(ch)) { |
||||
stream.eatWhile(operatorChars); |
||||
return null; |
||||
} else { |
||||
stream.eatWhile(/[_\w\d]/); |
||||
if (stream.eat(":")) { |
||||
stream.eatWhile(/[\w\d_\-]/); |
||||
return "atom"; |
||||
} |
||||
var word = stream.current(); |
||||
if (funcs.test(word)) return "builtin"; |
||||
if (preds.test(word)) return "def"; |
||||
if (keywords.test(word)) return "keyword"; |
||||
return "variable"; |
||||
} |
||||
}; |
||||
var pushContext = function(state, type, col) { |
||||
return state.context = { |
||||
prev: state.context, |
||||
indent: state.indent, |
||||
col: col, |
||||
type: type |
||||
}; |
||||
}; |
||||
var popContext = function(state) { |
||||
state.indent = state.context.indent; |
||||
return state.context = state.context.prev; |
||||
}; |
||||
var indentUnit = config.indentUnit; |
||||
var curPunc; |
||||
var funcs = wordRegexp(["abs", "acos", "allShortestPaths", "asin", "atan", "atan2", "avg", "ceil", "coalesce", "collect", "cos", "cot", "count", "degrees", "e", "endnode", "exp", "extract", "filter", "floor", "haversin", "head", "id", "keys", "labels", "last", "left", "length", "log", "log10", "lower", "ltrim", "max", "min", "node", "nodes", "percentileCont", "percentileDisc", "pi", "radians", "rand", "range", "reduce", "rel", "relationship", "relationships", "replace", "reverse", "right", "round", "rtrim", "shortestPath", "sign", "sin", "size", "split", "sqrt", "startnode", "stdev", "stdevp", "str", "substring", "sum", "tail", "tan", "timestamp", "toFloat", "toInt", "toString", "trim", "type", "upper"]); |
||||
var preds = wordRegexp(["all", "and", "any", "contains", "exists", "has", "in", "none", "not", "or", "single", "xor"]); |
||||
var keywords = wordRegexp(["as", "asc", "ascending", "assert", "by", "case", "commit", "constraint", "create", "csv", "cypher", "delete", "desc", "descending", "detach", "distinct", "drop", "else", "end", "ends", "explain", "false", "fieldterminator", "foreach", "from", "headers", "in", "index", "is", "join", "limit", "load", "match", "merge", "null", "on", "optional", "order", "periodic", "profile", "remove", "return", "scan", "set", "skip", "start", "starts", "then", "true", "union", "unique", "unwind", "using", "when", "where", "with"]); |
||||
var operatorChars = /[*+\-<>=&|~%^]/; |
||||
|
||||
return { |
||||
startState: function(/*base*/) { |
||||
return { |
||||
tokenize: tokenBase, |
||||
context: null, |
||||
indent: 0, |
||||
col: 0 |
||||
}; |
||||
}, |
||||
token: function(stream, state) { |
||||
if (stream.sol()) { |
||||
if (state.context && (state.context.align == null)) { |
||||
state.context.align = false; |
||||
} |
||||
state.indent = stream.indentation(); |
||||
} |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
var style = state.tokenize(stream, state); |
||||
if (style !== "comment" && state.context && (state.context.align == null) && state.context.type !== "pattern") { |
||||
state.context.align = true; |
||||
} |
||||
if (curPunc === "(") { |
||||
pushContext(state, ")", stream.column()); |
||||
} else if (curPunc === "[") { |
||||
pushContext(state, "]", stream.column()); |
||||
} else if (curPunc === "{") { |
||||
pushContext(state, "}", stream.column()); |
||||
} else if (/[\]\}\)]/.test(curPunc)) { |
||||
while (state.context && state.context.type === "pattern") { |
||||
popContext(state); |
||||
} |
||||
if (state.context && curPunc === state.context.type) { |
||||
popContext(state); |
||||
} |
||||
} else if (curPunc === "." && state.context && state.context.type === "pattern") { |
||||
popContext(state); |
||||
} else if (/atom|string|variable/.test(style) && state.context) { |
||||
if (/[\}\]]/.test(state.context.type)) { |
||||
pushContext(state, "pattern", stream.column()); |
||||
} else if (state.context.type === "pattern" && !state.context.align) { |
||||
state.context.align = true; |
||||
state.context.col = stream.column(); |
||||
} |
||||
} |
||||
return style; |
||||
}, |
||||
indent: function(state, textAfter) { |
||||
var firstChar = textAfter && textAfter.charAt(0); |
||||
var context = state.context; |
||||
if (/[\]\}]/.test(firstChar)) { |
||||
while (context && context.type === "pattern") { |
||||
context = context.prev; |
||||
} |
||||
} |
||||
var closing = context && firstChar === context.type; |
||||
if (!context) return 0; |
||||
if (context.type === "keywords") return CodeMirror.commands.newlineAndIndent; |
||||
if (context.align) return context.col + (closing ? 0 : 1); |
||||
return context.indent + (closing ? 0 : indentUnit); |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.modeExtensions["cypher"] = { |
||||
autoFormatLineBreaks: function(text) { |
||||
var i, lines, reProcessedPortion; |
||||
var lines = text.split("\n"); |
||||
var reProcessedPortion = /\s+\b(return|where|order by|match|with|skip|limit|create|delete|set)\b\s/g; |
||||
for (var i = 0; i < lines.length; i++) |
||||
lines[i] = lines[i].replace(reProcessedPortion, " \n$1 ").trim(); |
||||
return lines.join("\n"); |
||||
} |
||||
}; |
||||
|
||||
CodeMirror.defineMIME("application/x-cypher-query", "cypher"); |
||||
|
||||
}); |
@ -0,0 +1,63 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Cypher Mode for CodeMirror</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css" /> |
||||
<link rel="stylesheet" href="../../theme/neo.css" /> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="cypher.js"></script> |
||||
<style> |
||||
.CodeMirror { |
||||
border-top: 1px solid black; |
||||
border-bottom: 1px solid black; |
||||
} |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Cypher Mode for CodeMirror</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Cypher Mode for CodeMirror</h2> |
||||
<form> |
||||
<textarea id="code" name="code">// Cypher Mode for CodeMirror, using the neo theme |
||||
MATCH (joe { name: 'Joe' })-[:knows*2..2]-(friend_of_friend) |
||||
WHERE NOT (joe)-[:knows]-(friend_of_friend) |
||||
RETURN friend_of_friend.name, COUNT(*) |
||||
ORDER BY COUNT(*) DESC , friend_of_friend.name |
||||
</textarea> |
||||
</form> |
||||
<p><strong>MIME types defined:</strong> |
||||
<code><a href="?mime=application/x-cypher-query">application/x-cypher-query</a></code> |
||||
</p> |
||||
<script> |
||||
window.onload = function() { |
||||
var mime = 'application/x-cypher-query'; |
||||
// get mime type |
||||
if (window.location.href.indexOf('mime=') > -1) { |
||||
mime = window.location.href.substr(window.location.href.indexOf('mime=') + 5); |
||||
} |
||||
window.editor = CodeMirror.fromTextArea(document.getElementById('code'), { |
||||
mode: mime, |
||||
indentWithTabs: true, |
||||
smartIndent: true, |
||||
lineNumbers: true, |
||||
matchBrackets : true, |
||||
autofocus: true, |
||||
theme: 'neo' |
||||
}); |
||||
}; |
||||
</script> |
||||
|
||||
</article> |
@ -0,0 +1,218 @@ |
||||
// 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("d", function(config, parserConfig) { |
||||
var indentUnit = config.indentUnit, |
||||
statementIndentUnit = parserConfig.statementIndentUnit || indentUnit, |
||||
keywords = parserConfig.keywords || {}, |
||||
builtin = parserConfig.builtin || {}, |
||||
blockKeywords = parserConfig.blockKeywords || {}, |
||||
atoms = parserConfig.atoms || {}, |
||||
hooks = parserConfig.hooks || {}, |
||||
multiLineStrings = parserConfig.multiLineStrings; |
||||
var isOperatorChar = /[+\-*&%=<>!?|\/]/; |
||||
|
||||
var curPunc; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (hooks[ch]) { |
||||
var result = hooks[ch](stream, state); |
||||
if (result !== false) return result; |
||||
} |
||||
if (ch == '"' || ch == "'" || ch == "`") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (/[\[\]{}\(\),;\:\.]/.test(ch)) { |
||||
curPunc = ch; |
||||
return null; |
||||
} |
||||
if (/\d/.test(ch)) { |
||||
stream.eatWhile(/[\w\.]/); |
||||
return "number"; |
||||
} |
||||
if (ch == "/") { |
||||
if (stream.eat("+")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenNestedComment(stream, state); |
||||
} |
||||
if (stream.eat("*")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenComment(stream, state); |
||||
} |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
var cur = stream.current(); |
||||
if (keywords.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "keyword"; |
||||
} |
||||
if (builtin.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "builtin"; |
||||
} |
||||
if (atoms.propertyIsEnumerable(cur)) return "atom"; |
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) {end = true; break;} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !(escaped || multiLineStrings)) |
||||
state.tokenize = null; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
function tokenNestedComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "+"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
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) { |
||||
var indent = state.indented; |
||||
if (state.context && state.context.type == "statement") |
||||
indent = state.context.indented; |
||||
return state.context = new Context(indent, 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: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), |
||||
indented: 0, |
||||
startOfLine: true |
||||
}; |
||||
}, |
||||
|
||||
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; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
curPunc = null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment" || style == "meta") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
if ((curPunc == ";" || curPunc == ":" || curPunc == ",") && ctx.type == "statement") popContext(state); |
||||
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") && curPunc != ';') || (ctx.type == "statement" && curPunc == "newstatement")) |
||||
pushContext(state, stream.column(), "statement"); |
||||
state.startOfLine = false; |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != tokenBase && state.tokenize != null) return CodeMirror.Pass; |
||||
var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); |
||||
if (ctx.type == "statement" && firstChar == "}") ctx = ctx.prev; |
||||
var closing = firstChar == ctx.type; |
||||
if (ctx.type == "statement") return ctx.indented + (firstChar == "{" ? 0 : statementIndentUnit); |
||||
else if (ctx.align) return ctx.column + (closing ? 0 : 1); |
||||
else return ctx.indented + (closing ? 0 : indentUnit); |
||||
}, |
||||
|
||||
electricChars: "{}" |
||||
}; |
||||
}); |
||||
|
||||
function words(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
|
||||
var blockKeywords = "body catch class do else enum for foreach foreach_reverse if in interface mixin " + |
||||
"out scope struct switch try union unittest version while with"; |
||||
|
||||
CodeMirror.defineMIME("text/x-d", { |
||||
name: "d", |
||||
keywords: words("abstract alias align asm assert auto break case cast cdouble cent cfloat const continue " + |
||||
"debug default delegate delete deprecated export extern final finally function goto immutable " + |
||||
"import inout invariant is lazy macro module new nothrow override package pragma private " + |
||||
"protected public pure ref return shared short static super synchronized template this " + |
||||
"throw typedef typeid typeof volatile __FILE__ __LINE__ __gshared __traits __vector __parameters " + |
||||
blockKeywords), |
||||
blockKeywords: words(blockKeywords), |
||||
builtin: words("bool byte char creal dchar double float idouble ifloat int ireal long real short ubyte " + |
||||
"ucent uint ulong ushort wchar wstring void size_t sizediff_t"), |
||||
atoms: words("exit failure success true false null"), |
||||
hooks: { |
||||
"@": function(stream, _state) { |
||||
stream.eatWhile(/[\w\$_]/); |
||||
return "meta"; |
||||
} |
||||
} |
||||
}); |
||||
|
||||
}); |
@ -0,0 +1,273 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: D mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="d.js"></script> |
||||
<style>.CodeMirror {border: 2px inset #dee;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">D</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>D mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* D demo code // copied from phobos/sd/metastrings.d */ |
||||
// Written in the D programming language. |
||||
|
||||
/** |
||||
Templates with which to do compile-time manipulation of strings. |
||||
|
||||
Macros: |
||||
WIKI = Phobos/StdMetastrings |
||||
|
||||
Copyright: Copyright Digital Mars 2007 - 2009. |
||||
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>. |
||||
Authors: $(WEB digitalmars.com, Walter Bright), |
||||
Don Clugston |
||||
Source: $(PHOBOSSRC std/_metastrings.d) |
||||
*/ |
||||
/* |
||||
Copyright Digital Mars 2007 - 2009. |
||||
Distributed under the Boost Software License, Version 1.0. |
||||
(See accompanying file LICENSE_1_0.txt or copy at |
||||
http://www.boost.org/LICENSE_1_0.txt) |
||||
*/ |
||||
module std.metastrings; |
||||
|
||||
/** |
||||
Formats constants into a string at compile time. Analogous to $(XREF |
||||
string,format). |
||||
|
||||
Parameters: |
||||
|
||||
A = tuple of constants, which can be strings, characters, or integral |
||||
values. |
||||
|
||||
Formats: |
||||
* The formats supported are %s for strings, and %% |
||||
* for the % character. |
||||
Example: |
||||
--- |
||||
import std.metastrings; |
||||
import std.stdio; |
||||
|
||||
void main() |
||||
{ |
||||
string s = Format!("Arg %s = %s", "foo", 27); |
||||
writefln(s); // "Arg foo = 27" |
||||
} |
||||
* --- |
||||
*/ |
||||
|
||||
template Format(A...) |
||||
{ |
||||
static if (A.length == 0) |
||||
enum Format = ""; |
||||
else static if (is(typeof(A[0]) : const(char)[])) |
||||
enum Format = FormatString!(A[0], A[1..$]); |
||||
else |
||||
enum Format = toStringNow!(A[0]) ~ Format!(A[1..$]); |
||||
} |
||||
|
||||
template FormatString(const(char)[] F, A...) |
||||
{ |
||||
static if (F.length == 0) |
||||
enum FormatString = Format!(A); |
||||
else static if (F.length == 1) |
||||
enum FormatString = F[0] ~ Format!(A); |
||||
else static if (F[0..2] == "%s") |
||||
enum FormatString |
||||
= toStringNow!(A[0]) ~ FormatString!(F[2..$],A[1..$]); |
||||
else static if (F[0..2] == "%%") |
||||
enum FormatString = "%" ~ FormatString!(F[2..$],A); |
||||
else |
||||
{ |
||||
static assert(F[0] != '%', "unrecognized format %" ~ F[1]); |
||||
enum FormatString = F[0] ~ FormatString!(F[1..$],A); |
||||
} |
||||
} |
||||
|
||||
unittest |
||||
{ |
||||
auto s = Format!("hel%slo", "world", -138, 'c', true); |
||||
assert(s == "helworldlo-138ctrue", "[" ~ s ~ "]"); |
||||
} |
||||
|
||||
/** |
||||
* Convert constant argument to a string. |
||||
*/ |
||||
|
||||
template toStringNow(ulong v) |
||||
{ |
||||
static if (v < 10) |
||||
enum toStringNow = "" ~ cast(char)(v + '0'); |
||||
else |
||||
enum toStringNow = toStringNow!(v / 10) ~ toStringNow!(v % 10); |
||||
} |
||||
|
||||
unittest |
||||
{ |
||||
static assert(toStringNow!(1uL << 62) == "4611686018427387904"); |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(long v) |
||||
{ |
||||
static if (v < 0) |
||||
enum toStringNow = "-" ~ toStringNow!(cast(ulong) -v); |
||||
else |
||||
enum toStringNow = toStringNow!(cast(ulong) v); |
||||
} |
||||
|
||||
unittest |
||||
{ |
||||
static assert(toStringNow!(0x100000000) == "4294967296"); |
||||
static assert(toStringNow!(-138L) == "-138"); |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(uint U) |
||||
{ |
||||
enum toStringNow = toStringNow!(cast(ulong)U); |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(int I) |
||||
{ |
||||
enum toStringNow = toStringNow!(cast(long)I); |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(bool B) |
||||
{ |
||||
enum toStringNow = B ? "true" : "false"; |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(string S) |
||||
{ |
||||
enum toStringNow = S; |
||||
} |
||||
|
||||
/// ditto |
||||
template toStringNow(char C) |
||||
{ |
||||
enum toStringNow = "" ~ C; |
||||
} |
||||
|
||||
|
||||
/******** |
||||
* Parse unsigned integer literal from the start of string s. |
||||
* returns: |
||||
* .value = the integer literal as a string, |
||||
* .rest = the string following the integer literal |
||||
* Otherwise: |
||||
* .value = null, |
||||
* .rest = s |
||||
*/ |
||||
|
||||
template parseUinteger(const(char)[] s) |
||||
{ |
||||
static if (s.length == 0) |
||||
{ |
||||
enum value = ""; |
||||
enum rest = ""; |
||||
} |
||||
else static if (s[0] >= '0' && s[0] <= '9') |
||||
{ |
||||
enum value = s[0] ~ parseUinteger!(s[1..$]).value; |
||||
enum rest = parseUinteger!(s[1..$]).rest; |
||||
} |
||||
else |
||||
{ |
||||
enum value = ""; |
||||
enum rest = s; |
||||
} |
||||
} |
||||
|
||||
/******** |
||||
Parse integer literal optionally preceded by $(D '-') from the start |
||||
of string $(D s). |
||||
|
||||
Returns: |
||||
.value = the integer literal as a string, |
||||
.rest = the string following the integer literal |
||||
|
||||
Otherwise: |
||||
.value = null, |
||||
.rest = s |
||||
*/ |
||||
|
||||
template parseInteger(const(char)[] s) |
||||
{ |
||||
static if (s.length == 0) |
||||
{ |
||||
enum value = ""; |
||||
enum rest = ""; |
||||
} |
||||
else static if (s[0] >= '0' && s[0] <= '9') |
||||
{ |
||||
enum value = s[0] ~ parseUinteger!(s[1..$]).value; |
||||
enum rest = parseUinteger!(s[1..$]).rest; |
||||
} |
||||
else static if (s.length >= 2 && |
||||
s[0] == '-' && s[1] >= '0' && s[1] <= '9') |
||||
{ |
||||
enum value = s[0..2] ~ parseUinteger!(s[2..$]).value; |
||||
enum rest = parseUinteger!(s[2..$]).rest; |
||||
} |
||||
else |
||||
{ |
||||
enum value = ""; |
||||
enum rest = s; |
||||
} |
||||
} |
||||
|
||||
unittest |
||||
{ |
||||
assert(parseUinteger!("1234abc").value == "1234"); |
||||
assert(parseUinteger!("1234abc").rest == "abc"); |
||||
assert(parseInteger!("-1234abc").value == "-1234"); |
||||
assert(parseInteger!("-1234abc").rest == "abc"); |
||||
} |
||||
|
||||
/** |
||||
Deprecated aliases held for backward compatibility. |
||||
*/ |
||||
deprecated alias toStringNow ToString; |
||||
/// Ditto |
||||
deprecated alias parseUinteger ParseUinteger; |
||||
/// Ditto |
||||
deprecated alias parseUinteger ParseInteger; |
||||
|
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
indentUnit: 4, |
||||
mode: "text/x-d" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Simple mode that handle D-Syntax (<a href="http://www.dlang.org">DLang Homepage</a>).</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-d</code> |
||||
.</p> |
||||
</article> |
@ -0,0 +1,157 @@ |
||||
// 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"), require("../clike/clike")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../clike/clike"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
var keywords = ("this super static final const abstract class extends external factory " + |
||||
"implements get native operator set typedef with enum throw rethrow " + |
||||
"assert break case continue default in return new deferred async await " + |
||||
"try catch finally do else for if switch while import library export " + |
||||
"part of show hide is as").split(" "); |
||||
var blockKeywords = "try catch finally do else for if switch while".split(" "); |
||||
var atoms = "true false null".split(" "); |
||||
var builtins = "void bool num int double dynamic var String".split(" "); |
||||
|
||||
function set(words) { |
||||
var obj = {}; |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
|
||||
function pushInterpolationStack(state) { |
||||
(state.interpolationStack || (state.interpolationStack = [])).push(state.tokenize); |
||||
} |
||||
|
||||
function popInterpolationStack(state) { |
||||
return (state.interpolationStack || (state.interpolationStack = [])).pop(); |
||||
} |
||||
|
||||
function sizeInterpolationStack(state) { |
||||
return state.interpolationStack ? state.interpolationStack.length : 0; |
||||
} |
||||
|
||||
CodeMirror.defineMIME("application/dart", { |
||||
name: "clike", |
||||
keywords: set(keywords), |
||||
blockKeywords: set(blockKeywords), |
||||
builtin: set(builtins), |
||||
atoms: set(atoms), |
||||
hooks: { |
||||
"@": function(stream) { |
||||
stream.eatWhile(/[\w\$_\.]/); |
||||
return "meta"; |
||||
}, |
||||
|
||||
// custom string handling to deal with triple-quoted strings and string interpolation
|
||||
"'": function(stream, state) { |
||||
return tokenString("'", stream, state, false); |
||||
}, |
||||
"\"": function(stream, state) { |
||||
return tokenString("\"", stream, state, false); |
||||
}, |
||||
"r": function(stream, state) { |
||||
var peek = stream.peek(); |
||||
if (peek == "'" || peek == "\"") { |
||||
return tokenString(stream.next(), stream, state, true); |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
"}": function(_stream, state) { |
||||
// "}" is end of interpolation, if interpolation stack is non-empty
|
||||
if (sizeInterpolationStack(state) > 0) { |
||||
state.tokenize = popInterpolationStack(state); |
||||
return null; |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
"/": function(stream, state) { |
||||
if (!stream.eat("*")) return false |
||||
state.tokenize = tokenNestedComment(1) |
||||
return state.tokenize(stream, state) |
||||
} |
||||
} |
||||
}); |
||||
|
||||
function tokenString(quote, stream, state, raw) { |
||||
var tripleQuoted = false; |
||||
if (stream.eat(quote)) { |
||||
if (stream.eat(quote)) tripleQuoted = true; |
||||
else return "string"; //empty string
|
||||
} |
||||
function tokenStringHelper(stream, state) { |
||||
var escaped = false; |
||||
while (!stream.eol()) { |
||||
if (!raw && !escaped && stream.peek() == "$") { |
||||
pushInterpolationStack(state); |
||||
state.tokenize = tokenInterpolation; |
||||
return "string"; |
||||
} |
||||
var next = stream.next(); |
||||
if (next == quote && !escaped && (!tripleQuoted || stream.match(quote + quote))) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
escaped = !raw && !escaped && next == "\\"; |
||||
} |
||||
return "string"; |
||||
} |
||||
state.tokenize = tokenStringHelper; |
||||
return tokenStringHelper(stream, state); |
||||
} |
||||
|
||||
function tokenInterpolation(stream, state) { |
||||
stream.eat("$"); |
||||
if (stream.eat("{")) { |
||||
// let clike handle the content of ${...},
|
||||
// we take over again when "}" appears (see hooks).
|
||||
state.tokenize = null; |
||||
} else { |
||||
state.tokenize = tokenInterpolationIdentifier; |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
function tokenInterpolationIdentifier(stream, state) { |
||||
stream.eatWhile(/[\w_]/); |
||||
state.tokenize = popInterpolationStack(state); |
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenNestedComment(depth) { |
||||
return function (stream, state) { |
||||
var ch |
||||
while (ch = stream.next()) { |
||||
if (ch == "*" && stream.eat("/")) { |
||||
if (depth == 1) { |
||||
state.tokenize = null |
||||
break |
||||
} else { |
||||
state.tokenize = tokenNestedComment(depth - 1) |
||||
return state.tokenize(stream, state) |
||||
} |
||||
} else if (ch == "/" && stream.eat("*")) { |
||||
state.tokenize = tokenNestedComment(depth + 1) |
||||
return state.tokenize(stream, state) |
||||
} |
||||
} |
||||
return "comment" |
||||
} |
||||
} |
||||
|
||||
CodeMirror.registerHelper("hintWords", "application/dart", keywords.concat(atoms).concat(builtins)); |
||||
|
||||
// This is needed to make loading through meta.js work.
|
||||
CodeMirror.defineMode("dart", function(conf) { |
||||
return CodeMirror.getMode(conf, "application/dart"); |
||||
}, "clike"); |
||||
}); |
@ -0,0 +1,71 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Dart mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../clike/clike.js"></script> |
||||
<script src="dart.js"></script> |
||||
<style>.CodeMirror {border: 1px solid #dee;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Dart</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Dart mode</h2> |
||||
<form> |
||||
<textarea id="code" name="code"> |
||||
import 'dart:math' show Random; |
||||
|
||||
void main() { |
||||
print(new Die(n: 12).roll()); |
||||
} |
||||
|
||||
// Define a class. |
||||
class Die { |
||||
// Define a class variable. |
||||
static Random shaker = new Random(); |
||||
|
||||
// Define instance variables. |
||||
int sides, value; |
||||
|
||||
// Define a method using shorthand syntax. |
||||
String toString() => '$value'; |
||||
|
||||
// Define a constructor. |
||||
Die({int n: 6}) { |
||||
if (4 <= n && n <= 20) { |
||||
sides = n; |
||||
} else { |
||||
// Support for errors and exceptions. |
||||
throw new ArgumentError(/* */); |
||||
} |
||||
} |
||||
|
||||
// Define an instance method. |
||||
int roll() { |
||||
return value = shaker.nextInt(sides) + 1; |
||||
} |
||||
} |
||||
</textarea> |
||||
</form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "application/dart" |
||||
}); |
||||
</script> |
||||
|
||||
</article> |
@ -0,0 +1,47 @@ |
||||
// 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("diff", function() { |
||||
|
||||
var TOKEN_NAMES = { |
||||
'+': 'positive', |
||||
'-': 'negative', |
||||
'@': 'meta' |
||||
}; |
||||
|
||||
return { |
||||
token: function(stream) { |
||||
var tw_pos = stream.string.search(/[\t ]+?$/); |
||||
|
||||
if (!stream.sol() || tw_pos === 0) { |
||||
stream.skipToEnd(); |
||||
return ("error " + ( |
||||
TOKEN_NAMES[stream.string.charAt(0)] || '')).replace(/ $/, ''); |
||||
} |
||||
|
||||
var token_name = TOKEN_NAMES[stream.peek()] || stream.skipToEnd(); |
||||
|
||||
if (tw_pos === -1) { |
||||
stream.skipToEnd(); |
||||
} else { |
||||
stream.pos = tw_pos; |
||||
} |
||||
|
||||
return token_name; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-diff", "diff"); |
||||
|
||||
}); |
@ -0,0 +1,117 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Diff mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="diff.js"></script> |
||||
<style> |
||||
.CodeMirror {border-top: 1px solid #ddd; border-bottom: 1px solid #ddd;} |
||||
span.cm-meta {color: #a0b !important;} |
||||
span.cm-error { background-color: black; opacity: 0.4;} |
||||
span.cm-error.cm-string { background-color: red; } |
||||
span.cm-error.cm-tag { background-color: #2b2; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Diff</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Diff mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
diff --git a/index.html b/index.html |
||||
index c1d9156..7764744 100644 |
||||
--- a/index.html |
||||
+++ b/index.html |
||||
@@ -95,7 +95,8 @@ StringStream.prototype = { |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
- autoMatchBrackets: true |
||||
+ autoMatchBrackets: true, |
||||
+ onGutterClick: function(x){console.log(x);} |
||||
}); |
||||
</script> |
||||
</body> |
||||
diff --git a/lib/codemirror.js b/lib/codemirror.js |
||||
index 04646a9..9a39cc7 100644 |
||||
--- a/lib/codemirror.js |
||||
+++ b/lib/codemirror.js |
||||
@@ -399,10 +399,16 @@ var CodeMirror = (function() { |
||||
} |
||||
|
||||
function onMouseDown(e) { |
||||
- var start = posFromMouse(e), last = start; |
||||
+ var start = posFromMouse(e), last = start, target = e.target(); |
||||
if (!start) return; |
||||
setCursor(start.line, start.ch, false); |
||||
if (e.button() != 1) return; |
||||
+ if (target.parentNode == gutter) { |
||||
+ if (options.onGutterClick) |
||||
+ options.onGutterClick(indexOf(gutter.childNodes, target) + showingFrom); |
||||
+ return; |
||||
+ } |
||||
+ |
||||
if (!focused) onFocus(); |
||||
|
||||
e.stop(); |
||||
@@ -808,7 +814,7 @@ var CodeMirror = (function() { |
||||
for (var i = showingFrom; i < showingTo; ++i) { |
||||
var marker = lines[i].gutterMarker; |
||||
if (marker) html.push('<div class="' + marker.style + '">' + htmlEscape(marker.text) + '</div>'); |
||||
- else html.push("<div>" + (options.lineNumbers ? i + 1 : "\u00a0") + "</div>"); |
||||
+ else html.push("<div>" + (options.lineNumbers ? i + options.firstLineNumber : "\u00a0") + "</div>"); |
||||
} |
||||
gutter.style.display = "none"; // TODO test whether this actually helps |
||||
gutter.innerHTML = html.join(""); |
||||
@@ -1371,10 +1377,8 @@ var CodeMirror = (function() { |
||||
if (option == "parser") setParser(value); |
||||
else if (option === "lineNumbers") setLineNumbers(value); |
||||
else if (option === "gutter") setGutter(value); |
||||
- else if (option === "readOnly") options.readOnly = value; |
||||
- else if (option === "indentUnit") {options.indentUnit = indentUnit = value; setParser(options.parser);} |
||||
- else if (/^(?:enterMode|tabMode|indentWithTabs|readOnly|autoMatchBrackets|undoDepth)$/.test(option)) options[option] = value; |
||||
- else throw new Error("Can't set option " + option); |
||||
+ else if (option === "indentUnit") {options.indentUnit = value; setParser(options.parser);} |
||||
+ else options[option] = value; |
||||
}, |
||||
cursorCoords: cursorCoords, |
||||
undo: operation(undo), |
||||
@@ -1402,7 +1406,8 @@ var CodeMirror = (function() { |
||||
replaceRange: operation(replaceRange), |
||||
|
||||
operation: function(f){return operation(f)();}, |
||||
- refresh: function(){updateDisplay([{from: 0, to: lines.length}]);} |
||||
+ refresh: function(){updateDisplay([{from: 0, to: lines.length}]);}, |
||||
+ getInputField: function(){return input;} |
||||
}; |
||||
return instance; |
||||
} |
||||
@@ -1420,6 +1425,7 @@ var CodeMirror = (function() { |
||||
readOnly: false, |
||||
onChange: null, |
||||
onCursorActivity: null, |
||||
+ onGutterClick: null, |
||||
autoMatchBrackets: false, |
||||
workTime: 200, |
||||
workDelay: 300, |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-diff</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,356 @@ |
||||
// 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"), require("../htmlmixed/htmlmixed"), |
||||
require("../../addon/mode/overlay")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../htmlmixed/htmlmixed", |
||||
"../../addon/mode/overlay"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
CodeMirror.defineMode("django:inner", function() { |
||||
var keywords = ["block", "endblock", "for", "endfor", "true", "false", "filter", "endfilter", |
||||
"loop", "none", "self", "super", "if", "elif", "endif", "as", "else", "import", |
||||
"with", "endwith", "without", "context", "ifequal", "endifequal", "ifnotequal", |
||||
"endifnotequal", "extends", "include", "load", "comment", "endcomment", |
||||
"empty", "url", "static", "trans", "blocktrans", "endblocktrans", "now", |
||||
"regroup", "lorem", "ifchanged", "endifchanged", "firstof", "debug", "cycle", |
||||
"csrf_token", "autoescape", "endautoescape", "spaceless", "endspaceless", |
||||
"ssi", "templatetag", "verbatim", "endverbatim", "widthratio"], |
||||
filters = ["add", "addslashes", "capfirst", "center", "cut", "date", |
||||
"default", "default_if_none", "dictsort", |
||||
"dictsortreversed", "divisibleby", "escape", "escapejs", |
||||
"filesizeformat", "first", "floatformat", "force_escape", |
||||
"get_digit", "iriencode", "join", "last", "length", |
||||
"length_is", "linebreaks", "linebreaksbr", "linenumbers", |
||||
"ljust", "lower", "make_list", "phone2numeric", "pluralize", |
||||
"pprint", "random", "removetags", "rjust", "safe", |
||||
"safeseq", "slice", "slugify", "stringformat", "striptags", |
||||
"time", "timesince", "timeuntil", "title", "truncatechars", |
||||
"truncatechars_html", "truncatewords", "truncatewords_html", |
||||
"unordered_list", "upper", "urlencode", "urlize", |
||||
"urlizetrunc", "wordcount", "wordwrap", "yesno"], |
||||
operators = ["==", "!=", "<", ">", "<=", ">="], |
||||
wordOperators = ["in", "not", "or", "and"]; |
||||
|
||||
keywords = new RegExp("^\\b(" + keywords.join("|") + ")\\b"); |
||||
filters = new RegExp("^\\b(" + filters.join("|") + ")\\b"); |
||||
operators = new RegExp("^\\b(" + operators.join("|") + ")\\b"); |
||||
wordOperators = new RegExp("^\\b(" + wordOperators.join("|") + ")\\b"); |
||||
|
||||
// We have to return "null" instead of null, in order to avoid string
|
||||
// styling as the default, when using Django templates inside HTML
|
||||
// element attributes
|
||||
function tokenBase (stream, state) { |
||||
// Attempt to identify a variable, template or comment tag respectively
|
||||
if (stream.match("{{")) { |
||||
state.tokenize = inVariable; |
||||
return "tag"; |
||||
} else if (stream.match("{%")) { |
||||
state.tokenize = inTag; |
||||
return "tag"; |
||||
} else if (stream.match("{#")) { |
||||
state.tokenize = inComment; |
||||
return "comment"; |
||||
} |
||||
|
||||
// Ignore completely any stream series that do not match the
|
||||
// Django template opening tags.
|
||||
while (stream.next() != null && !stream.match(/\{[{%#]/, false)) {} |
||||
return null; |
||||
} |
||||
|
||||
// A string can be included in either single or double quotes (this is
|
||||
// the delimiter). Mark everything as a string until the start delimiter
|
||||
// occurs again.
|
||||
function inString (delimiter, previousTokenizer) { |
||||
return function (stream, state) { |
||||
if (!state.escapeNext && stream.eat(delimiter)) { |
||||
state.tokenize = previousTokenizer; |
||||
} else { |
||||
if (state.escapeNext) { |
||||
state.escapeNext = false; |
||||
} |
||||
|
||||
var ch = stream.next(); |
||||
|
||||
// Take into account the backslash for escaping characters, such as
|
||||
// the string delimiter.
|
||||
if (ch == "\\") { |
||||
state.escapeNext = true; |
||||
} |
||||
} |
||||
|
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
// Apply Django template variable syntax highlighting
|
||||
function inVariable (stream, state) { |
||||
// Attempt to match a dot that precedes a property
|
||||
if (state.waitDot) { |
||||
state.waitDot = false; |
||||
|
||||
if (stream.peek() != ".") { |
||||
return "null"; |
||||
} |
||||
|
||||
// Dot followed by a non-word character should be considered an error.
|
||||
if (stream.match(/\.\W+/)) { |
||||
return "error"; |
||||
} else if (stream.eat(".")) { |
||||
state.waitProperty = true; |
||||
return "null"; |
||||
} else { |
||||
throw Error ("Unexpected error while waiting for property."); |
||||
} |
||||
} |
||||
|
||||
// Attempt to match a pipe that precedes a filter
|
||||
if (state.waitPipe) { |
||||
state.waitPipe = false; |
||||
|
||||
if (stream.peek() != "|") { |
||||
return "null"; |
||||
} |
||||
|
||||
// Pipe followed by a non-word character should be considered an error.
|
||||
if (stream.match(/\.\W+/)) { |
||||
return "error"; |
||||
} else if (stream.eat("|")) { |
||||
state.waitFilter = true; |
||||
return "null"; |
||||
} else { |
||||
throw Error ("Unexpected error while waiting for filter."); |
||||
} |
||||
} |
||||
|
||||
// Highlight properties
|
||||
if (state.waitProperty) { |
||||
state.waitProperty = false; |
||||
if (stream.match(/\b(\w+)\b/)) { |
||||
state.waitDot = true; // A property can be followed by another property
|
||||
state.waitPipe = true; // A property can be followed by a filter
|
||||
return "property"; |
||||
} |
||||
} |
||||
|
||||
// Highlight filters
|
||||
if (state.waitFilter) { |
||||
state.waitFilter = false; |
||||
if (stream.match(filters)) { |
||||
return "variable-2"; |
||||
} |
||||
} |
||||
|
||||
// Ignore all white spaces
|
||||
if (stream.eatSpace()) { |
||||
state.waitProperty = false; |
||||
return "null"; |
||||
} |
||||
|
||||
// Identify numbers
|
||||
if (stream.match(/\b\d+(\.\d+)?\b/)) { |
||||
return "number"; |
||||
} |
||||
|
||||
// Identify strings
|
||||
if (stream.match("'")) { |
||||
state.tokenize = inString("'", state.tokenize); |
||||
return "string"; |
||||
} else if (stream.match('"')) { |
||||
state.tokenize = inString('"', state.tokenize); |
||||
return "string"; |
||||
} |
||||
|
||||
// Attempt to find the variable
|
||||
if (stream.match(/\b(\w+)\b/) && !state.foundVariable) { |
||||
state.waitDot = true; |
||||
state.waitPipe = true; // A property can be followed by a filter
|
||||
return "variable"; |
||||
} |
||||
|
||||
// If found closing tag reset
|
||||
if (stream.match("}}")) { |
||||
state.waitProperty = null; |
||||
state.waitFilter = null; |
||||
state.waitDot = null; |
||||
state.waitPipe = null; |
||||
state.tokenize = tokenBase; |
||||
return "tag"; |
||||
} |
||||
|
||||
// If nothing was found, advance to the next character
|
||||
stream.next(); |
||||
return "null"; |
||||
} |
||||
|
||||
function inTag (stream, state) { |
||||
// Attempt to match a dot that precedes a property
|
||||
if (state.waitDot) { |
||||
state.waitDot = false; |
||||
|
||||
if (stream.peek() != ".") { |
||||
return "null"; |
||||
} |
||||
|
||||
// Dot followed by a non-word character should be considered an error.
|
||||
if (stream.match(/\.\W+/)) { |
||||
return "error"; |
||||
} else if (stream.eat(".")) { |
||||
state.waitProperty = true; |
||||
return "null"; |
||||
} else { |
||||
throw Error ("Unexpected error while waiting for property."); |
||||
} |
||||
} |
||||
|
||||
// Attempt to match a pipe that precedes a filter
|
||||
if (state.waitPipe) { |
||||
state.waitPipe = false; |
||||
|
||||
if (stream.peek() != "|") { |
||||
return "null"; |
||||
} |
||||
|
||||
// Pipe followed by a non-word character should be considered an error.
|
||||
if (stream.match(/\.\W+/)) { |
||||
return "error"; |
||||
} else if (stream.eat("|")) { |
||||
state.waitFilter = true; |
||||
return "null"; |
||||
} else { |
||||
throw Error ("Unexpected error while waiting for filter."); |
||||
} |
||||
} |
||||
|
||||
// Highlight properties
|
||||
if (state.waitProperty) { |
||||
state.waitProperty = false; |
||||
if (stream.match(/\b(\w+)\b/)) { |
||||
state.waitDot = true; // A property can be followed by another property
|
||||
state.waitPipe = true; // A property can be followed by a filter
|
||||
return "property"; |
||||
} |
||||
} |
||||
|
||||
// Highlight filters
|
||||
if (state.waitFilter) { |
||||
state.waitFilter = false; |
||||
if (stream.match(filters)) { |
||||
return "variable-2"; |
||||
} |
||||
} |
||||
|
||||
// Ignore all white spaces
|
||||
if (stream.eatSpace()) { |
||||
state.waitProperty = false; |
||||
return "null"; |
||||
} |
||||
|
||||
// Identify numbers
|
||||
if (stream.match(/\b\d+(\.\d+)?\b/)) { |
||||
return "number"; |
||||
} |
||||
|
||||
// Identify strings
|
||||
if (stream.match("'")) { |
||||
state.tokenize = inString("'", state.tokenize); |
||||
return "string"; |
||||
} else if (stream.match('"')) { |
||||
state.tokenize = inString('"', state.tokenize); |
||||
return "string"; |
||||
} |
||||
|
||||
// Attempt to match an operator
|
||||
if (stream.match(operators)) { |
||||
return "operator"; |
||||
} |
||||
|
||||
// Attempt to match a word operator
|
||||
if (stream.match(wordOperators)) { |
||||
return "keyword"; |
||||
} |
||||
|
||||
// Attempt to match a keyword
|
||||
var keywordMatch = stream.match(keywords); |
||||
if (keywordMatch) { |
||||
if (keywordMatch[0] == "comment") { |
||||
state.blockCommentTag = true; |
||||
} |
||||
return "keyword"; |
||||
} |
||||
|
||||
// Attempt to match a variable
|
||||
if (stream.match(/\b(\w+)\b/)) { |
||||
state.waitDot = true; |
||||
state.waitPipe = true; // A property can be followed by a filter
|
||||
return "variable"; |
||||
} |
||||
|
||||
// If found closing tag reset
|
||||
if (stream.match("%}")) { |
||||
state.waitProperty = null; |
||||
state.waitFilter = null; |
||||
state.waitDot = null; |
||||
state.waitPipe = null; |
||||
// If the tag that closes is a block comment tag, we want to mark the
|
||||
// following code as comment, until the tag closes.
|
||||
if (state.blockCommentTag) { |
||||
state.blockCommentTag = false; // Release the "lock"
|
||||
state.tokenize = inBlockComment; |
||||
} else { |
||||
state.tokenize = tokenBase; |
||||
} |
||||
return "tag"; |
||||
} |
||||
|
||||
// If nothing was found, advance to the next character
|
||||
stream.next(); |
||||
return "null"; |
||||
} |
||||
|
||||
// Mark everything as comment inside the tag and the tag itself.
|
||||
function inComment (stream, state) { |
||||
if (stream.match(/^.*?#\}/)) state.tokenize = tokenBase |
||||
else stream.skipToEnd() |
||||
return "comment"; |
||||
} |
||||
|
||||
// Mark everything as a comment until the `blockcomment` tag closes.
|
||||
function inBlockComment (stream, state) { |
||||
if (stream.match(/\{%\s*endcomment\s*%\}/, false)) { |
||||
state.tokenize = inTag; |
||||
stream.match("{%"); |
||||
return "tag"; |
||||
} else { |
||||
stream.next(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
|
||||
return { |
||||
startState: function () { |
||||
return {tokenize: tokenBase}; |
||||
}, |
||||
token: function (stream, state) { |
||||
return state.tokenize(stream, state); |
||||
}, |
||||
blockCommentStart: "{% comment %}", |
||||
blockCommentEnd: "{% endcomment %}" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMode("django", function(config) { |
||||
var htmlBase = CodeMirror.getMode(config, "text/html"); |
||||
var djangoInner = CodeMirror.getMode(config, "django:inner"); |
||||
return CodeMirror.overlayMode(htmlBase, djangoInner); |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-django", "django"); |
||||
}); |
@ -0,0 +1,73 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Django template mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/mdn-like.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/mode/overlay.js"></script> |
||||
<script src="../xml/xml.js"></script> |
||||
<script src="../htmlmixed/htmlmixed.js"></script> |
||||
<script src="django.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Django</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Django template mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
<!doctype html> |
||||
<html> |
||||
<head> |
||||
<title>My Django web application</title> |
||||
</head> |
||||
<body> |
||||
<h1> |
||||
{{ page.title|capfirst }} |
||||
</h1> |
||||
<ul class="my-list"> |
||||
{# traverse a list of items and produce links to their views. #} |
||||
{% for item in items %} |
||||
<li> |
||||
<a href="{% url 'item_view' item.name|slugify %}"> |
||||
{{ item.name }} |
||||
</a> |
||||
</li> |
||||
{% empty %} |
||||
<li>You have no items in your list.</li> |
||||
{% endfor %} |
||||
</ul> |
||||
{% comment "this is a forgotten footer" %} |
||||
<footer></footer> |
||||
{% endcomment %} |
||||
</body> |
||||
</html> |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "django", |
||||
indentUnit: 2, |
||||
indentWithTabs: true, |
||||
theme: "mdn-like" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Mode for HTML with embedded Django template markup.</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-django</code></p> |
||||
</article> |
@ -0,0 +1,79 @@ |
||||
// 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"), require("../../addon/mode/simple")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../../addon/mode/simple"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
// Collect all Dockerfile directives
|
||||
var instructions = ["from", "maintainer", "run", "cmd", "expose", "env", |
||||
"add", "copy", "entrypoint", "volume", "user", |
||||
"workdir", "onbuild"], |
||||
instructionRegex = "(" + instructions.join('|') + ")", |
||||
instructionOnlyLine = new RegExp(instructionRegex + "\\s*$", "i"), |
||||
instructionWithArguments = new RegExp(instructionRegex + "(\\s+)", "i"); |
||||
|
||||
CodeMirror.defineSimpleMode("dockerfile", { |
||||
start: [ |
||||
// Block comment: This is a line starting with a comment
|
||||
{ |
||||
regex: /#.*$/, |
||||
token: "comment" |
||||
}, |
||||
// Highlight an instruction without any arguments (for convenience)
|
||||
{ |
||||
regex: instructionOnlyLine, |
||||
token: "variable-2" |
||||
}, |
||||
// Highlight an instruction followed by arguments
|
||||
{ |
||||
regex: instructionWithArguments, |
||||
token: ["variable-2", null], |
||||
next: "arguments" |
||||
}, |
||||
{ |
||||
regex: /./, |
||||
token: null |
||||
} |
||||
], |
||||
arguments: [ |
||||
{ |
||||
// Line comment without instruction arguments is an error
|
||||
regex: /#.*$/, |
||||
token: "error", |
||||
next: "start" |
||||
}, |
||||
{ |
||||
regex: /[^#]+\\$/, |
||||
token: null |
||||
}, |
||||
{ |
||||
// Match everything except for the inline comment
|
||||
regex: /[^#]+/, |
||||
token: null, |
||||
next: "start" |
||||
}, |
||||
{ |
||||
regex: /$/, |
||||
token: null, |
||||
next: "start" |
||||
}, |
||||
// Fail safe return to start
|
||||
{ |
||||
token: null, |
||||
next: "start" |
||||
} |
||||
], |
||||
meta: { |
||||
lineComment: "#" |
||||
} |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-dockerfile", "dockerfile"); |
||||
}); |
@ -0,0 +1,73 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Dockerfile mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/mode/simple.js"></script> |
||||
<script src="dockerfile.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Dockerfile</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Dockerfile mode</h2> |
||||
<form><textarea id="code" name="code"># Install Ghost blogging platform and run development environment |
||||
# |
||||
# VERSION 1.0.0 |
||||
|
||||
FROM ubuntu:12.10 |
||||
MAINTAINER Amer Grgic "amer@livebyt.es" |
||||
WORKDIR /data/ghost |
||||
|
||||
# Install dependencies for nginx installation |
||||
RUN apt-get update |
||||
RUN apt-get install -y python g++ make software-properties-common --force-yes |
||||
RUN add-apt-repository ppa:chris-lea/node.js |
||||
RUN apt-get update |
||||
# Install unzip |
||||
RUN apt-get install -y unzip |
||||
# Install curl |
||||
RUN apt-get install -y curl |
||||
# Install nodejs & npm |
||||
RUN apt-get install -y rlwrap |
||||
RUN apt-get install -y nodejs |
||||
# Download Ghost v0.4.1 |
||||
RUN curl -L https://ghost.org/zip/ghost-latest.zip -o /tmp/ghost.zip |
||||
# Unzip Ghost zip to /data/ghost |
||||
RUN unzip -uo /tmp/ghost.zip -d /data/ghost |
||||
# Add custom config js to /data/ghost |
||||
ADD ./config.example.js /data/ghost/config.js |
||||
# Install Ghost with NPM |
||||
RUN cd /data/ghost/ && npm install --production |
||||
# Expose port 2368 |
||||
EXPOSE 2368 |
||||
# Run Ghost |
||||
CMD ["npm","start"] |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "dockerfile" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Dockerfile syntax highlighting for CodeMirror. Depends on |
||||
the <a href="../../demo/simplemode.html">simplemode</a> addon.</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-dockerfile</code></p> |
||||
</article> |
@ -0,0 +1,142 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/* |
||||
DTD mode |
||||
Ported to CodeMirror by Peter Kroon <plakroon@gmail.com> |
||||
Report bugs/issues here: https://github.com/codemirror/CodeMirror/issues
|
||||
GitHub: @peterkroon |
||||
*/ |
||||
|
||||
(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("dtd", function(config) { |
||||
var indentUnit = config.indentUnit, type; |
||||
function ret(style, tp) {type = tp; return style;} |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
|
||||
if (ch == "<" && stream.eat("!") ) { |
||||
if (stream.eatWhile(/[\-]/)) { |
||||
state.tokenize = tokenSGMLComment; |
||||
return tokenSGMLComment(stream, state); |
||||
} else if (stream.eatWhile(/[\w]/)) return ret("keyword", "doindent"); |
||||
} else if (ch == "<" && stream.eat("?")) { //xml declaration
|
||||
state.tokenize = inBlock("meta", "?>"); |
||||
return ret("meta", ch); |
||||
} else if (ch == "#" && stream.eatWhile(/[\w]/)) return ret("atom", "tag"); |
||||
else if (ch == "|") return ret("keyword", "seperator"); |
||||
else if (ch.match(/[\(\)\[\]\-\.,\+\?>]/)) return ret(null, ch);//if(ch === ">") return ret(null, "endtag"); else
|
||||
else if (ch.match(/[\[\]]/)) return ret("rule", ch); |
||||
else if (ch == "\"" || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} else if (stream.eatWhile(/[a-zA-Z\?\+\d]/)) { |
||||
var sc = stream.current(); |
||||
if( sc.substr(sc.length-1,sc.length).match(/\?|\+/) !== null )stream.backUp(1); |
||||
return ret("tag", "tag"); |
||||
} else if (ch == "%" || ch == "*" ) return ret("number", "number"); |
||||
else { |
||||
stream.eatWhile(/[\w\\\-_%.{,]/); |
||||
return ret(null, null); |
||||
} |
||||
} |
||||
|
||||
function tokenSGMLComment(stream, state) { |
||||
var dashes = 0, ch; |
||||
while ((ch = stream.next()) != null) { |
||||
if (dashes >= 2 && ch == ">") { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
dashes = (ch == "-") ? dashes + 1 : 0; |
||||
} |
||||
return ret("comment", "comment"); |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, ch; |
||||
while ((ch = stream.next()) != null) { |
||||
if (ch == quote && !escaped) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
escaped = !escaped && ch == "\\"; |
||||
} |
||||
return ret("string", "tag"); |
||||
}; |
||||
} |
||||
|
||||
function inBlock(style, terminator) { |
||||
return function(stream, state) { |
||||
while (!stream.eol()) { |
||||
if (stream.match(terminator)) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
stream.next(); |
||||
} |
||||
return style; |
||||
}; |
||||
} |
||||
|
||||
return { |
||||
startState: function(base) { |
||||
return {tokenize: tokenBase, |
||||
baseIndent: base || 0, |
||||
stack: []}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (stream.eatSpace()) return null; |
||||
var style = state.tokenize(stream, state); |
||||
|
||||
var context = state.stack[state.stack.length-1]; |
||||
if (stream.current() == "[" || type === "doindent" || type == "[") state.stack.push("rule"); |
||||
else if (type === "endtag") state.stack[state.stack.length-1] = "endtag"; |
||||
else if (stream.current() == "]" || type == "]" || (type == ">" && context == "rule")) state.stack.pop(); |
||||
else if (type == "[") state.stack.push("["); |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
var n = state.stack.length; |
||||
|
||||
if( textAfter.match(/\]\s+|\]/) )n=n-1; |
||||
else if(textAfter.substr(textAfter.length-1, textAfter.length) === ">"){ |
||||
if(textAfter.substr(0,1) === "<") {} |
||||
else if( type == "doindent" && textAfter.length > 1 ) {} |
||||
else if( type == "doindent")n--; |
||||
else if( type == ">" && textAfter.length > 1) {} |
||||
else if( type == "tag" && textAfter !== ">") {} |
||||
else if( type == "tag" && state.stack[state.stack.length-1] == "rule")n--; |
||||
else if( type == "tag")n++; |
||||
else if( textAfter === ">" && state.stack[state.stack.length-1] == "rule" && type === ">")n--; |
||||
else if( textAfter === ">" && state.stack[state.stack.length-1] == "rule") {} |
||||
else if( textAfter.substr(0,1) !== "<" && textAfter.substr(0,1) === ">" )n=n-1; |
||||
else if( textAfter === ">") {} |
||||
else n=n-1; |
||||
//over rule them all
|
||||
if(type == null || type == "]")n--; |
||||
} |
||||
|
||||
return state.baseIndent + n * indentUnit; |
||||
}, |
||||
|
||||
electricChars: "]>" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("application/xml-dtd", "dtd"); |
||||
|
||||
}); |
@ -0,0 +1,89 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: DTD mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="dtd.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">DTD</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>DTD mode</h2> |
||||
<form><textarea id="code" name="code"><?xml version="1.0" encoding="UTF-8"?> |
||||
|
||||
<!ATTLIST title |
||||
xmlns CDATA #FIXED "http://docbook.org/ns/docbook" |
||||
role CDATA #IMPLIED |
||||
%db.common.attributes; |
||||
%db.common.linking.attributes; |
||||
> |
||||
|
||||
<!-- |
||||
Try: http://docbook.org/xml/5.0/dtd/docbook.dtd |
||||
--> |
||||
|
||||
<!DOCTYPE xsl:stylesheet |
||||
[ |
||||
<!ENTITY nbsp "&#160;"> |
||||
<!ENTITY copy "&#169;"> |
||||
<!ENTITY reg "&#174;"> |
||||
<!ENTITY trade "&#8482;"> |
||||
<!ENTITY mdash "&#8212;"> |
||||
<!ENTITY ldquo "&#8220;"> |
||||
<!ENTITY rdquo "&#8221;"> |
||||
<!ENTITY pound "&#163;"> |
||||
<!ENTITY yen "&#165;"> |
||||
<!ENTITY euro "&#8364;"> |
||||
<!ENTITY mathml "http://www.w3.org/1998/Math/MathML"> |
||||
] |
||||
> |
||||
|
||||
<!ELEMENT title (#PCDATA|inlinemediaobject|remark|superscript|subscript|xref|link|olink|anchor|biblioref|alt|annotation|indexterm|abbrev|acronym|date|emphasis|footnote|footnoteref|foreignphrase|phrase|quote|wordasword|firstterm|glossterm|coref|trademark|productnumber|productname|database|application|hardware|citation|citerefentry|citetitle|citebiblioid|author|person|personname|org|orgname|editor|jobtitle|replaceable|package|parameter|termdef|nonterminal|systemitem|option|optional|property|inlineequation|tag|markup|token|symbol|literal|code|constant|email|uri|guiicon|guibutton|guimenuitem|guimenu|guisubmenu|guilabel|menuchoice|mousebutton|keycombo|keycap|keycode|keysym|shortcut|accel|prompt|envar|filename|command|computeroutput|userinput|function|varname|returnvalue|type|classname|exceptionname|interfacename|methodname|modifier|initializer|ooclass|ooexception|oointerface|errorcode|errortext|errorname|errortype)*> |
||||
|
||||
<!ENTITY % db.common.attributes " |
||||
xml:id ID #IMPLIED |
||||
version CDATA #IMPLIED |
||||
xml:lang CDATA #IMPLIED |
||||
xml:base CDATA #IMPLIED |
||||
remap CDATA #IMPLIED |
||||
xreflabel CDATA #IMPLIED |
||||
revisionflag (changed|added|deleted|off) #IMPLIED |
||||
dir (ltr|rtl|lro|rlo) #IMPLIED |
||||
arch CDATA #IMPLIED |
||||
audience CDATA #IMPLIED |
||||
condition CDATA #IMPLIED |
||||
conformance CDATA #IMPLIED |
||||
os CDATA #IMPLIED |
||||
revision CDATA #IMPLIED |
||||
security CDATA #IMPLIED |
||||
userlevel CDATA #IMPLIED |
||||
vendor CDATA #IMPLIED |
||||
wordsize CDATA #IMPLIED |
||||
annotations CDATA #IMPLIED |
||||
|
||||
"></textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: {name: "dtd", alignCDATA: true}, |
||||
lineNumbers: true, |
||||
lineWrapping: true |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>application/xml-dtd</code>.</p> |
||||
</article> |
@ -0,0 +1,344 @@ |
||||
// 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("dylan", function(_config) { |
||||
// Words
|
||||
var words = { |
||||
// Words that introduce unnamed definitions like "define interface"
|
||||
unnamedDefinition: ["interface"], |
||||
|
||||
// Words that introduce simple named definitions like "define library"
|
||||
namedDefinition: ["module", "library", "macro", |
||||
"C-struct", "C-union", |
||||
"C-function", "C-callable-wrapper" |
||||
], |
||||
|
||||
// Words that introduce type definitions like "define class".
|
||||
// These are also parameterized like "define method" and are
|
||||
// appended to otherParameterizedDefinitionWords
|
||||
typeParameterizedDefinition: ["class", "C-subtype", "C-mapped-subtype"], |
||||
|
||||
// Words that introduce trickier definitions like "define method".
|
||||
// These require special definitions to be added to startExpressions
|
||||
otherParameterizedDefinition: ["method", "function", |
||||
"C-variable", "C-address" |
||||
], |
||||
|
||||
// Words that introduce module constant definitions.
|
||||
// These must also be simple definitions and are
|
||||
// appended to otherSimpleDefinitionWords
|
||||
constantSimpleDefinition: ["constant"], |
||||
|
||||
// Words that introduce module variable definitions.
|
||||
// These must also be simple definitions and are
|
||||
// appended to otherSimpleDefinitionWords
|
||||
variableSimpleDefinition: ["variable"], |
||||
|
||||
// Other words that introduce simple definitions
|
||||
// (without implicit bodies).
|
||||
otherSimpleDefinition: ["generic", "domain", |
||||
"C-pointer-type", |
||||
"table" |
||||
], |
||||
|
||||
// Words that begin statements with implicit bodies.
|
||||
statement: ["if", "block", "begin", "method", "case", |
||||
"for", "select", "when", "unless", "until", |
||||
"while", "iterate", "profiling", "dynamic-bind" |
||||
], |
||||
|
||||
// Patterns that act as separators in compound statements.
|
||||
// This may include any general pattern that must be indented
|
||||
// specially.
|
||||
separator: ["finally", "exception", "cleanup", "else", |
||||
"elseif", "afterwards" |
||||
], |
||||
|
||||
// Keywords that do not require special indentation handling,
|
||||
// but which should be highlighted
|
||||
other: ["above", "below", "by", "from", "handler", "in", |
||||
"instance", "let", "local", "otherwise", "slot", |
||||
"subclass", "then", "to", "keyed-by", "virtual" |
||||
], |
||||
|
||||
// Condition signaling function calls
|
||||
signalingCalls: ["signal", "error", "cerror", |
||||
"break", "check-type", "abort" |
||||
] |
||||
}; |
||||
|
||||
words["otherDefinition"] = |
||||
words["unnamedDefinition"] |
||||
.concat(words["namedDefinition"]) |
||||
.concat(words["otherParameterizedDefinition"]); |
||||
|
||||
words["definition"] = |
||||
words["typeParameterizedDefinition"] |
||||
.concat(words["otherDefinition"]); |
||||
|
||||
words["parameterizedDefinition"] = |
||||
words["typeParameterizedDefinition"] |
||||
.concat(words["otherParameterizedDefinition"]); |
||||
|
||||
words["simpleDefinition"] = |
||||
words["constantSimpleDefinition"] |
||||
.concat(words["variableSimpleDefinition"]) |
||||
.concat(words["otherSimpleDefinition"]); |
||||
|
||||
words["keyword"] = |
||||
words["statement"] |
||||
.concat(words["separator"]) |
||||
.concat(words["other"]); |
||||
|
||||
// Patterns
|
||||
var symbolPattern = "[-_a-zA-Z?!*@<>$%]+"; |
||||
var symbol = new RegExp("^" + symbolPattern); |
||||
var patterns = { |
||||
// Symbols with special syntax
|
||||
symbolKeyword: symbolPattern + ":", |
||||
symbolClass: "<" + symbolPattern + ">", |
||||
symbolGlobal: "\\*" + symbolPattern + "\\*", |
||||
symbolConstant: "\\$" + symbolPattern |
||||
}; |
||||
var patternStyles = { |
||||
symbolKeyword: "atom", |
||||
symbolClass: "tag", |
||||
symbolGlobal: "variable-2", |
||||
symbolConstant: "variable-3" |
||||
}; |
||||
|
||||
// Compile all patterns to regular expressions
|
||||
for (var patternName in patterns) |
||||
if (patterns.hasOwnProperty(patternName)) |
||||
patterns[patternName] = new RegExp("^" + patterns[patternName]); |
||||
|
||||
// Names beginning "with-" and "without-" are commonly
|
||||
// used as statement macro
|
||||
patterns["keyword"] = [/^with(?:out)?-[-_a-zA-Z?!*@<>$%]+/]; |
||||
|
||||
var styles = {}; |
||||
styles["keyword"] = "keyword"; |
||||
styles["definition"] = "def"; |
||||
styles["simpleDefinition"] = "def"; |
||||
styles["signalingCalls"] = "builtin"; |
||||
|
||||
// protected words lookup table
|
||||
var wordLookup = {}; |
||||
var styleLookup = {}; |
||||
|
||||
[ |
||||
"keyword", |
||||
"definition", |
||||
"simpleDefinition", |
||||
"signalingCalls" |
||||
].forEach(function(type) { |
||||
words[type].forEach(function(word) { |
||||
wordLookup[word] = type; |
||||
styleLookup[word] = styles[type]; |
||||
}); |
||||
}); |
||||
|
||||
|
||||
function chain(stream, state, f) { |
||||
state.tokenize = f; |
||||
return f(stream, state); |
||||
} |
||||
|
||||
function tokenBase(stream, state) { |
||||
// String
|
||||
var ch = stream.peek(); |
||||
if (ch == "'" || ch == '"') { |
||||
stream.next(); |
||||
return chain(stream, state, tokenString(ch, "string")); |
||||
} |
||||
// Comment
|
||||
else if (ch == "/") { |
||||
stream.next(); |
||||
if (stream.eat("*")) { |
||||
return chain(stream, state, tokenComment); |
||||
} else if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
stream.backUp(1); |
||||
} |
||||
// Decimal
|
||||
else if (/[+\-\d\.]/.test(ch)) { |
||||
if (stream.match(/^[+-]?[0-9]*\.[0-9]*([esdx][+-]?[0-9]+)?/i) || |
||||
stream.match(/^[+-]?[0-9]+([esdx][+-]?[0-9]+)/i) || |
||||
stream.match(/^[+-]?\d+/)) { |
||||
return "number"; |
||||
} |
||||
} |
||||
// Hash
|
||||
else if (ch == "#") { |
||||
stream.next(); |
||||
// Symbol with string syntax
|
||||
ch = stream.peek(); |
||||
if (ch == '"') { |
||||
stream.next(); |
||||
return chain(stream, state, tokenString('"', "string")); |
||||
} |
||||
// Binary number
|
||||
else if (ch == "b") { |
||||
stream.next(); |
||||
stream.eatWhile(/[01]/); |
||||
return "number"; |
||||
} |
||||
// Hex number
|
||||
else if (ch == "x") { |
||||
stream.next(); |
||||
stream.eatWhile(/[\da-f]/i); |
||||
return "number"; |
||||
} |
||||
// Octal number
|
||||
else if (ch == "o") { |
||||
stream.next(); |
||||
stream.eatWhile(/[0-7]/); |
||||
return "number"; |
||||
} |
||||
// Token concatenation in macros
|
||||
else if (ch == '#') { |
||||
stream.next(); |
||||
return "punctuation"; |
||||
} |
||||
// Sequence literals
|
||||
else if ((ch == '[') || (ch == '(')) { |
||||
stream.next(); |
||||
return "bracket"; |
||||
// Hash symbol
|
||||
} else if (stream.match(/f|t|all-keys|include|key|next|rest/i)) { |
||||
return "atom"; |
||||
} else { |
||||
stream.eatWhile(/[-a-zA-Z]/); |
||||
return "error"; |
||||
} |
||||
} else if (ch == "~") { |
||||
stream.next(); |
||||
ch = stream.peek(); |
||||
if (ch == "=") { |
||||
stream.next(); |
||||
ch = stream.peek(); |
||||
if (ch == "=") { |
||||
stream.next(); |
||||
return "operator"; |
||||
} |
||||
return "operator"; |
||||
} |
||||
return "operator"; |
||||
} else if (ch == ":") { |
||||
stream.next(); |
||||
ch = stream.peek(); |
||||
if (ch == "=") { |
||||
stream.next(); |
||||
return "operator"; |
||||
} else if (ch == ":") { |
||||
stream.next(); |
||||
return "punctuation"; |
||||
} |
||||
} else if ("[](){}".indexOf(ch) != -1) { |
||||
stream.next(); |
||||
return "bracket"; |
||||
} else if (".,".indexOf(ch) != -1) { |
||||
stream.next(); |
||||
return "punctuation"; |
||||
} else if (stream.match("end")) { |
||||
return "keyword"; |
||||
} |
||||
for (var name in patterns) { |
||||
if (patterns.hasOwnProperty(name)) { |
||||
var pattern = patterns[name]; |
||||
if ((pattern instanceof Array && pattern.some(function(p) { |
||||
return stream.match(p); |
||||
})) || stream.match(pattern)) |
||||
return patternStyles[name]; |
||||
} |
||||
} |
||||
if (/[+\-*\/^=<>&|]/.test(ch)) { |
||||
stream.next(); |
||||
return "operator"; |
||||
} |
||||
if (stream.match("define")) { |
||||
return "def"; |
||||
} else { |
||||
stream.eatWhile(/[\w\-]/); |
||||
// Keyword
|
||||
if (wordLookup[stream.current()]) { |
||||
return styleLookup[stream.current()]; |
||||
} else if (stream.current().match(symbol)) { |
||||
return "variable"; |
||||
} else { |
||||
stream.next(); |
||||
return "variable-2"; |
||||
} |
||||
} |
||||
} |
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, maybeNested = false, nestedCount = 0, ch; |
||||
while ((ch = stream.next())) { |
||||
if (ch == "/" && maybeEnd) { |
||||
if (nestedCount > 0) { |
||||
nestedCount--; |
||||
} else { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
} else if (ch == "*" && maybeNested) { |
||||
nestedCount++; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
maybeNested = (ch == "/"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
function tokenString(quote, style) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) { |
||||
end = true; |
||||
break; |
||||
} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !escaped) { |
||||
state.tokenize = tokenBase; |
||||
} |
||||
return style; |
||||
}; |
||||
} |
||||
|
||||
// Interface
|
||||
return { |
||||
startState: function() { |
||||
return { |
||||
tokenize: tokenBase, |
||||
currentIndent: 0 |
||||
}; |
||||
}, |
||||
token: function(stream, state) { |
||||
if (stream.eatSpace()) |
||||
return null; |
||||
var style = state.tokenize(stream, state); |
||||
return style; |
||||
}, |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-dylan", "dylan"); |
||||
|
||||
}); |
@ -0,0 +1,407 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Dylan mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="../../addon/comment/continuecomment.js"></script> |
||||
<script src="../../addon/comment/comment.js"></script> |
||||
<script src="dylan.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Dylan</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Dylan mode</h2> |
||||
|
||||
|
||||
<div><textarea id="code" name="code"> |
||||
Module: locators-internals |
||||
Synopsis: Abstract modeling of locations |
||||
Author: Andy Armstrong |
||||
Copyright: Original Code is Copyright (c) 1995-2004 Functional Objects, Inc. |
||||
All rights reserved. |
||||
License: See License.txt in this distribution for details. |
||||
Warranty: Distributed WITHOUT WARRANTY OF ANY KIND |
||||
|
||||
define open generic locator-server |
||||
(locator :: <locator>) => (server :: false-or(<server-locator>)); |
||||
define open generic locator-host |
||||
(locator :: <locator>) => (host :: false-or(<string>)); |
||||
define open generic locator-volume |
||||
(locator :: <locator>) => (volume :: false-or(<string>)); |
||||
define open generic locator-directory |
||||
(locator :: <locator>) => (directory :: false-or(<directory-locator>)); |
||||
define open generic locator-relative? |
||||
(locator :: <locator>) => (relative? :: <boolean>); |
||||
define open generic locator-path |
||||
(locator :: <locator>) => (path :: <sequence>); |
||||
define open generic locator-base |
||||
(locator :: <locator>) => (base :: false-or(<string>)); |
||||
define open generic locator-extension |
||||
(locator :: <locator>) => (extension :: false-or(<string>)); |
||||
|
||||
/// Locator classes |
||||
|
||||
define open abstract class <directory-locator> (<physical-locator>) |
||||
end class <directory-locator>; |
||||
|
||||
define open abstract class <file-locator> (<physical-locator>) |
||||
end class <file-locator>; |
||||
|
||||
define method as |
||||
(class == <directory-locator>, string :: <string>) |
||||
=> (locator :: <directory-locator>) |
||||
as(<native-directory-locator>, string) |
||||
end method as; |
||||
|
||||
define method make |
||||
(class == <directory-locator>, |
||||
#key server :: false-or(<server-locator>) = #f, |
||||
path :: <sequence> = #[], |
||||
relative? :: <boolean> = #f, |
||||
name :: false-or(<string>) = #f) |
||||
=> (locator :: <directory-locator>) |
||||
make(<native-directory-locator>, |
||||
server: server, |
||||
path: path, |
||||
relative?: relative?, |
||||
name: name) |
||||
end method make; |
||||
|
||||
define method as |
||||
(class == <file-locator>, string :: <string>) |
||||
=> (locator :: <file-locator>) |
||||
as(<native-file-locator>, string) |
||||
end method as; |
||||
|
||||
define method make |
||||
(class == <file-locator>, |
||||
#key directory :: false-or(<directory-locator>) = #f, |
||||
base :: false-or(<string>) = #f, |
||||
extension :: false-or(<string>) = #f, |
||||
name :: false-or(<string>) = #f) |
||||
=> (locator :: <file-locator>) |
||||
make(<native-file-locator>, |
||||
directory: directory, |
||||
base: base, |
||||
extension: extension, |
||||
name: name) |
||||
end method make; |
||||
|
||||
/// Locator coercion |
||||
|
||||
//---*** andrewa: This caching scheme doesn't work yet, so disable it. |
||||
define constant $cache-locators? = #f; |
||||
define constant $cache-locator-strings? = #f; |
||||
|
||||
define constant $locator-to-string-cache = make(<object-table>, weak: #"key"); |
||||
define constant $string-to-locator-cache = make(<string-table>, weak: #"value"); |
||||
|
||||
define open generic locator-as-string |
||||
(class :: subclass(<string>), locator :: <locator>) |
||||
=> (string :: <string>); |
||||
|
||||
define open generic string-as-locator |
||||
(class :: subclass(<locator>), string :: <string>) |
||||
=> (locator :: <locator>); |
||||
|
||||
define sealed sideways method as |
||||
(class :: subclass(<string>), locator :: <locator>) |
||||
=> (string :: <string>) |
||||
let string = element($locator-to-string-cache, locator, default: #f); |
||||
if (string) |
||||
as(class, string) |
||||
else |
||||
let string = locator-as-string(class, locator); |
||||
if ($cache-locator-strings?) |
||||
element($locator-to-string-cache, locator) := string; |
||||
else |
||||
string |
||||
end |
||||
end |
||||
end method as; |
||||
|
||||
define sealed sideways method as |
||||
(class :: subclass(<locator>), string :: <string>) |
||||
=> (locator :: <locator>) |
||||
let locator = element($string-to-locator-cache, string, default: #f); |
||||
if (instance?(locator, class)) |
||||
locator |
||||
else |
||||
let locator = string-as-locator(class, string); |
||||
if ($cache-locators?) |
||||
element($string-to-locator-cache, string) := locator; |
||||
else |
||||
locator |
||||
end |
||||
end |
||||
end method as; |
||||
|
||||
/// Locator conditions |
||||
|
||||
define class <locator-error> (<format-string-condition>, <error>) |
||||
end class <locator-error>; |
||||
|
||||
define function locator-error |
||||
(format-string :: <string>, #rest format-arguments) |
||||
error(make(<locator-error>, |
||||
format-string: format-string, |
||||
format-arguments: format-arguments)) |
||||
end function locator-error; |
||||
|
||||
/// Useful locator protocols |
||||
|
||||
define open generic locator-test |
||||
(locator :: <directory-locator>) => (test :: <function>); |
||||
|
||||
define method locator-test |
||||
(locator :: <directory-locator>) => (test :: <function>) |
||||
\= |
||||
end method locator-test; |
||||
|
||||
define open generic locator-might-have-links? |
||||
(locator :: <directory-locator>) => (links? :: <boolean>); |
||||
|
||||
define method locator-might-have-links? |
||||
(locator :: <directory-locator>) => (links? :: singleton(#f)) |
||||
#f |
||||
end method locator-might-have-links?; |
||||
|
||||
define method locator-relative? |
||||
(locator :: <file-locator>) => (relative? :: <boolean>) |
||||
let directory = locator.locator-directory; |
||||
~directory | directory.locator-relative? |
||||
end method locator-relative?; |
||||
|
||||
define method current-directory-locator? |
||||
(locator :: <directory-locator>) => (current-directory? :: <boolean>) |
||||
locator.locator-relative? |
||||
& locator.locator-path = #[#"self"] |
||||
end method current-directory-locator?; |
||||
|
||||
define method locator-directory |
||||
(locator :: <directory-locator>) => (parent :: false-or(<directory-locator>)) |
||||
let path = locator.locator-path; |
||||
unless (empty?(path)) |
||||
make(object-class(locator), |
||||
server: locator.locator-server, |
||||
path: copy-sequence(path, end: path.size - 1), |
||||
relative?: locator.locator-relative?) |
||||
end |
||||
end method locator-directory; |
||||
|
||||
/// Simplify locator |
||||
|
||||
define open generic simplify-locator |
||||
(locator :: <physical-locator>) |
||||
=> (simplified-locator :: <physical-locator>); |
||||
|
||||
define method simplify-locator |
||||
(locator :: <directory-locator>) |
||||
=> (simplified-locator :: <directory-locator>) |
||||
let path = locator.locator-path; |
||||
let relative? = locator.locator-relative?; |
||||
let resolve-parent? = ~locator.locator-might-have-links?; |
||||
let simplified-path |
||||
= simplify-path(path, |
||||
resolve-parent?: resolve-parent?, |
||||
relative?: relative?); |
||||
if (path ~= simplified-path) |
||||
make(object-class(locator), |
||||
server: locator.locator-server, |
||||
path: simplified-path, |
||||
relative?: locator.locator-relative?) |
||||
else |
||||
locator |
||||
end |
||||
end method simplify-locator; |
||||
|
||||
define method simplify-locator |
||||
(locator :: <file-locator>) => (simplified-locator :: <file-locator>) |
||||
let directory = locator.locator-directory; |
||||
let simplified-directory = directory & simplify-locator(directory); |
||||
if (directory ~= simplified-directory) |
||||
make(object-class(locator), |
||||
directory: simplified-directory, |
||||
base: locator.locator-base, |
||||
extension: locator.locator-extension) |
||||
else |
||||
locator |
||||
end |
||||
end method simplify-locator; |
||||
|
||||
/// Subdirectory locator |
||||
|
||||
define open generic subdirectory-locator |
||||
(locator :: <directory-locator>, #rest sub-path) |
||||
=> (subdirectory :: <directory-locator>); |
||||
|
||||
define method subdirectory-locator |
||||
(locator :: <directory-locator>, #rest sub-path) |
||||
=> (subdirectory :: <directory-locator>) |
||||
let old-path = locator.locator-path; |
||||
let new-path = concatenate-as(<simple-object-vector>, old-path, sub-path); |
||||
make(object-class(locator), |
||||
server: locator.locator-server, |
||||
path: new-path, |
||||
relative?: locator.locator-relative?) |
||||
end method subdirectory-locator; |
||||
|
||||
/// Relative locator |
||||
|
||||
define open generic relative-locator |
||||
(locator :: <physical-locator>, from-locator :: <physical-locator>) |
||||
=> (relative-locator :: <physical-locator>); |
||||
|
||||
define method relative-locator |
||||
(locator :: <directory-locator>, from-locator :: <directory-locator>) |
||||
=> (relative-locator :: <directory-locator>) |
||||
let path = locator.locator-path; |
||||
let from-path = from-locator.locator-path; |
||||
case |
||||
~locator.locator-relative? & from-locator.locator-relative? => |
||||
locator-error |
||||
("Cannot find relative path of absolute locator %= from relative locator %=", |
||||
locator, from-locator); |
||||
locator.locator-server ~= from-locator.locator-server => |
||||
locator; |
||||
path = from-path => |
||||
make(object-class(locator), |
||||
path: vector(#"self"), |
||||
relative?: #t); |
||||
otherwise => |
||||
make(object-class(locator), |
||||
path: relative-path(path, from-path, test: locator.locator-test), |
||||
relative?: #t); |
||||
end |
||||
end method relative-locator; |
||||
|
||||
define method relative-locator |
||||
(locator :: <file-locator>, from-directory :: <directory-locator>) |
||||
=> (relative-locator :: <file-locator>) |
||||
let directory = locator.locator-directory; |
||||
let relative-directory = directory & relative-locator(directory, from-directory); |
||||
if (relative-directory ~= directory) |
||||
simplify-locator |
||||
(make(object-class(locator), |
||||
directory: relative-directory, |
||||
base: locator.locator-base, |
||||
extension: locator.locator-extension)) |
||||
else |
||||
locator |
||||
end |
||||
end method relative-locator; |
||||
|
||||
define method relative-locator |
||||
(locator :: <physical-locator>, from-locator :: <file-locator>) |
||||
=> (relative-locator :: <physical-locator>) |
||||
let from-directory = from-locator.locator-directory; |
||||
case |
||||
from-directory => |
||||
relative-locator(locator, from-directory); |
||||
~locator.locator-relative? => |
||||
locator-error |
||||
("Cannot find relative path of absolute locator %= from relative locator %=", |
||||
locator, from-locator); |
||||
otherwise => |
||||
locator; |
||||
end |
||||
end method relative-locator; |
||||
|
||||
/// Merge locators |
||||
|
||||
define open generic merge-locators |
||||
(locator :: <physical-locator>, from-locator :: <physical-locator>) |
||||
=> (merged-locator :: <physical-locator>); |
||||
|
||||
/// Merge locators |
||||
|
||||
define method merge-locators |
||||
(locator :: <directory-locator>, from-locator :: <directory-locator>) |
||||
=> (merged-locator :: <directory-locator>) |
||||
if (locator.locator-relative?) |
||||
let path = concatenate(from-locator.locator-path, locator.locator-path); |
||||
simplify-locator |
||||
(make(object-class(locator), |
||||
server: from-locator.locator-server, |
||||
path: path, |
||||
relative?: from-locator.locator-relative?)) |
||||
else |
||||
locator |
||||
end |
||||
end method merge-locators; |
||||
|
||||
define method merge-locators |
||||
(locator :: <file-locator>, from-locator :: <directory-locator>) |
||||
=> (merged-locator :: <file-locator>) |
||||
let directory = locator.locator-directory; |
||||
let merged-directory |
||||
= if (directory) |
||||
merge-locators(directory, from-locator) |
||||
else |
||||
simplify-locator(from-locator) |
||||
end; |
||||
if (merged-directory ~= directory) |
||||
make(object-class(locator), |
||||
directory: merged-directory, |
||||
base: locator.locator-base, |
||||
extension: locator.locator-extension) |
||||
else |
||||
locator |
||||
end |
||||
end method merge-locators; |
||||
|
||||
define method merge-locators |
||||
(locator :: <physical-locator>, from-locator :: <file-locator>) |
||||
=> (merged-locator :: <physical-locator>) |
||||
let from-directory = from-locator.locator-directory; |
||||
if (from-directory) |
||||
merge-locators(locator, from-directory) |
||||
else |
||||
locator |
||||
end |
||||
end method merge-locators; |
||||
|
||||
/// Locator protocols |
||||
|
||||
define sideways method supports-open-locator? |
||||
(locator :: <file-locator>) => (openable? :: <boolean>) |
||||
~locator.locator-relative? |
||||
end method supports-open-locator?; |
||||
|
||||
define sideways method open-locator |
||||
(locator :: <file-locator>, #rest keywords, #key, #all-keys) |
||||
=> (stream :: <stream>) |
||||
apply(open-file-stream, locator, keywords) |
||||
end method open-locator; |
||||
</textarea></div> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: "text/x-dylan", |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
continueComments: "Enter", |
||||
extraKeys: {"Ctrl-Q": "toggleComment"}, |
||||
tabMode: "indent", |
||||
indentUnit: 2 |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-dylan</code>.</p> |
||||
</article> |
@ -0,0 +1,88 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({indentUnit: 2}, "dylan"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
MT('comments', |
||||
'[comment // This is a line comment]', |
||||
'[comment /* This is a block comment */]', |
||||
'[comment /* This is a multi]', |
||||
'[comment line comment]', |
||||
'[comment */]', |
||||
'[comment /* And this is a /*]', |
||||
'[comment /* nested */ comment */]'); |
||||
|
||||
MT('unary_operators', |
||||
'[operator -][variable a]', |
||||
'[operator -] [variable a]', |
||||
'[operator ~][variable a]', |
||||
'[operator ~] [variable a]'); |
||||
|
||||
MT('binary_operators', |
||||
'[variable a] [operator +] [variable b]', |
||||
'[variable a] [operator -] [variable b]', |
||||
'[variable a] [operator *] [variable b]', |
||||
'[variable a] [operator /] [variable b]', |
||||
'[variable a] [operator ^] [variable b]', |
||||
'[variable a] [operator =] [variable b]', |
||||
'[variable a] [operator ==] [variable b]', |
||||
'[variable a] [operator ~=] [variable b]', |
||||
'[variable a] [operator ~==] [variable b]', |
||||
'[variable a] [operator <] [variable b]', |
||||
'[variable a] [operator <=] [variable b]', |
||||
'[variable a] [operator >] [variable b]', |
||||
'[variable a] [operator >=] [variable b]', |
||||
'[variable a] [operator &] [variable b]', |
||||
'[variable a] [operator |] [variable b]', |
||||
'[variable a] [operator :=] [variable b]'); |
||||
|
||||
MT('integers', |
||||
'[number 1]', |
||||
'[number 123]', |
||||
'[number -123]', |
||||
'[number +456]', |
||||
'[number #b010]', |
||||
'[number #o073]', |
||||
'[number #xabcDEF123]'); |
||||
|
||||
MT('floats', |
||||
'[number .3]', |
||||
'[number -1.]', |
||||
'[number -2.335]', |
||||
'[number +3.78d1]', |
||||
'[number 3.78s-1]', |
||||
'[number -3.32e+5]'); |
||||
|
||||
MT('characters_and_strings', |
||||
"[string 'a']", |
||||
"[string '\\\\'']", |
||||
'[string ""]', |
||||
'[string "a"]', |
||||
'[string "abc def"]', |
||||
'[string "More escaped characters: \\\\\\\\ \\\\a \\\\b \\\\e \\\\f \\\\n \\\\r \\\\t \\\\0 ..."]'); |
||||
|
||||
MT('brackets', |
||||
'[bracket #[[]]]', |
||||
'[bracket #()]', |
||||
'[bracket #(][number 1][bracket )]', |
||||
'[bracket [[][number 1][punctuation ,] [number 3][bracket ]]]', |
||||
'[bracket ()]', |
||||
'[bracket {}]', |
||||
'[keyword if] [bracket (][variable foo][bracket )]', |
||||
'[bracket (][number 1][bracket )]', |
||||
'[bracket [[][number 1][bracket ]]]'); |
||||
|
||||
MT('hash_words', |
||||
'[punctuation ##]', |
||||
'[atom #f]', '[atom #F]', |
||||
'[atom #t]', '[atom #T]', |
||||
'[atom #all-keys]', |
||||
'[atom #include]', |
||||
'[atom #key]', |
||||
'[atom #next]', |
||||
'[atom #rest]', |
||||
'[string #"foo"]', |
||||
'[error #invalid]'); |
||||
})(); |
@ -0,0 +1,195 @@ |
||||
// 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"); |
||||
}); |
@ -0,0 +1,102 @@ |
||||
<!doctype html> |
||||
<html> |
||||
<head> |
||||
<title>CodeMirror: EBNF Mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../javascript/javascript.js"></script> |
||||
<script src="ebnf.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
</head> |
||||
<body> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">EBNF Mode</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>EBNF Mode (bracesMode setting = "javascript")</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* description: Parses end executes mathematical expressions. */ |
||||
|
||||
/* lexical grammar */ |
||||
%lex |
||||
|
||||
%% |
||||
\s+ /* skip whitespace */ |
||||
[0-9]+("."[0-9]+)?\b return 'NUMBER'; |
||||
"*" return '*'; |
||||
"/" return '/'; |
||||
"-" return '-'; |
||||
"+" return '+'; |
||||
"^" return '^'; |
||||
"(" return '('; |
||||
")" return ')'; |
||||
"PI" return 'PI'; |
||||
"E" return 'E'; |
||||
<<EOF>> return 'EOF'; |
||||
|
||||
/lex |
||||
|
||||
/* operator associations and precedence */ |
||||
|
||||
%left '+' '-' |
||||
%left '*' '/' |
||||
%left '^' |
||||
%left UMINUS |
||||
|
||||
%start expressions |
||||
|
||||
%% /* language grammar */ |
||||
|
||||
expressions |
||||
: e EOF |
||||
{print($1); return $1;} |
||||
; |
||||
|
||||
e |
||||
: e '+' e |
||||
{$$ = $1+$3;} |
||||
| e '-' e |
||||
{$$ = $1-$3;} |
||||
| e '*' e |
||||
{$$ = $1*$3;} |
||||
| e '/' e |
||||
{$$ = $1/$3;} |
||||
| e '^' e |
||||
{$$ = Math.pow($1, $3);} |
||||
| '-' e %prec UMINUS |
||||
{$$ = -$2;} |
||||
| '(' e ')' |
||||
{$$ = $2;} |
||||
| NUMBER |
||||
{$$ = Number(yytext);} |
||||
| E |
||||
{$$ = Math.E;} |
||||
| PI |
||||
{$$ = Math.PI;} |
||||
;</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: {name: "ebnf"}, |
||||
lineNumbers: true, |
||||
bracesMode: 'javascript' |
||||
}); |
||||
</script> |
||||
<h3>The EBNF Mode</h3> |
||||
<p> Created by <a href="https://github.com/robertleeplummerjr">Robert Plummer</a></p> |
||||
</article> |
||||
</body> |
||||
</html> |
@ -0,0 +1,206 @@ |
||||
// 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("ecl", function(config) { |
||||
|
||||
function words(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
|
||||
function metaHook(stream, state) { |
||||
if (!state.startOfLine) return false; |
||||
stream.skipToEnd(); |
||||
return "meta"; |
||||
} |
||||
|
||||
var indentUnit = config.indentUnit; |
||||
var keyword = words("abs acos allnodes ascii asin asstring atan atan2 ave case choose choosen choosesets clustersize combine correlation cos cosh count covariance cron dataset dedup define denormalize distribute distributed distribution ebcdic enth error evaluate event eventextra eventname exists exp failcode failmessage fetch fromunicode getisvalid global graph group hash hash32 hash64 hashcrc hashmd5 having if index intformat isvalid iterate join keyunicode length library limit ln local log loop map matched matchlength matchposition matchtext matchunicode max merge mergejoin min nolocal nonempty normalize parse pipe power preload process project pull random range rank ranked realformat recordof regexfind regexreplace regroup rejected rollup round roundup row rowdiff sample set sin sinh sizeof soapcall sort sorted sqrt stepped stored sum table tan tanh thisnode topn tounicode transfer trim truncate typeof ungroup unicodeorder variance which workunit xmldecode xmlencode xmltext xmlunicode"); |
||||
var variable = words("apply assert build buildindex evaluate fail keydiff keypatch loadxml nothor notify output parallel sequential soapcall wait"); |
||||
var variable_2 = words("__compressed__ all and any as atmost before beginc++ best between case const counter csv descend encrypt end endc++ endmacro except exclusive expire export extend false few first flat from full function group header heading hole ifblock import in interface joined keep keyed last left limit load local locale lookup macro many maxcount maxlength min skew module named nocase noroot noscan nosort not of only opt or outer overwrite packed partition penalty physicallength pipe quote record relationship repeat return right scan self separator service shared skew skip sql store terminator thor threshold token transform trim true type unicodeorder unsorted validate virtual whole wild within xml xpath"); |
||||
var variable_3 = words("ascii big_endian boolean data decimal ebcdic integer pattern qstring real record rule set of string token udecimal unicode unsigned varstring varunicode"); |
||||
var builtin = words("checkpoint deprecated failcode failmessage failure global independent onwarning persist priority recovery stored success wait when"); |
||||
var blockKeywords = words("catch class do else finally for if switch try while"); |
||||
var atoms = words("true false null"); |
||||
var hooks = {"#": metaHook}; |
||||
var isOperatorChar = /[+\-*&%=<>!?|\/]/; |
||||
|
||||
var curPunc; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (hooks[ch]) { |
||||
var result = hooks[ch](stream, state); |
||||
if (result !== false) return result; |
||||
} |
||||
if (ch == '"' || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (/[\[\]{}\(\),;\:\.]/.test(ch)) { |
||||
curPunc = ch; |
||||
return null; |
||||
} |
||||
if (/\d/.test(ch)) { |
||||
stream.eatWhile(/[\w\.]/); |
||||
return "number"; |
||||
} |
||||
if (ch == "/") { |
||||
if (stream.eat("*")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenComment(stream, state); |
||||
} |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_]/); |
||||
var cur = stream.current().toLowerCase(); |
||||
if (keyword.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "keyword"; |
||||
} else if (variable.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "variable"; |
||||
} else if (variable_2.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "variable-2"; |
||||
} else if (variable_3.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "variable-3"; |
||||
} else if (builtin.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
return "builtin"; |
||||
} else { //Data types are of from KEYWORD##
|
||||
var i = cur.length - 1; |
||||
while(i >= 0 && (!isNaN(cur[i]) || cur[i] == '_')) |
||||
--i; |
||||
|
||||
if (i > 0) { |
||||
var cur2 = cur.substr(0, i + 1); |
||||
if (variable_3.propertyIsEnumerable(cur2)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur2)) curPunc = "newstatement"; |
||||
return "variable-3"; |
||||
} |
||||
} |
||||
} |
||||
if (atoms.propertyIsEnumerable(cur)) return "atom"; |
||||
return null; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) {end = true; break;} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !escaped) |
||||
state.tokenize = tokenBase; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
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: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), |
||||
indented: 0, |
||||
startOfLine: true |
||||
}; |
||||
}, |
||||
|
||||
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; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
curPunc = null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment" || style == "meta") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
if ((curPunc == ";" || curPunc == ":") && ctx.type == "statement") popContext(state); |
||||
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; |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != tokenBase && state.tokenize != null) return 0; |
||||
var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); |
||||
if (ctx.type == "statement" && firstChar == "}") ctx = ctx.prev; |
||||
var closing = firstChar == ctx.type; |
||||
if (ctx.type == "statement") return ctx.indented + (firstChar == "{" ? 0 : indentUnit); |
||||
else if (ctx.align) return ctx.column + (closing ? 0 : 1); |
||||
else return ctx.indented + (closing ? 0 : indentUnit); |
||||
}, |
||||
|
||||
electricChars: "{}" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-ecl", "ecl"); |
||||
|
||||
}); |
@ -0,0 +1,52 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: ECL mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="ecl.js"></script> |
||||
<style>.CodeMirror {border: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">ECL</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>ECL mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
/* |
||||
sample useless code to demonstrate ecl syntax highlighting |
||||
this is a multiline comment! |
||||
*/ |
||||
|
||||
// this is a singleline comment! |
||||
|
||||
import ut; |
||||
r := |
||||
record |
||||
string22 s1 := '123'; |
||||
integer4 i1 := 123; |
||||
end; |
||||
#option('tmp', true); |
||||
d := dataset('tmp::qb', r, thor); |
||||
output(d); |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {}); |
||||
</script> |
||||
|
||||
<p>Based on CodeMirror's clike mode. For more information see <a href="http://hpccsystems.com">HPCC Systems</a> web site.</p> |
||||
<p><strong>MIME types defined:</strong> <code>text/x-ecl</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,160 @@ |
||||
// 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("eiffel", function() { |
||||
function wordObj(words) { |
||||
var o = {}; |
||||
for (var i = 0, e = words.length; i < e; ++i) o[words[i]] = true; |
||||
return o; |
||||
} |
||||
var keywords = wordObj([ |
||||
'note', |
||||
'across', |
||||
'when', |
||||
'variant', |
||||
'until', |
||||
'unique', |
||||
'undefine', |
||||
'then', |
||||
'strip', |
||||
'select', |
||||
'retry', |
||||
'rescue', |
||||
'require', |
||||
'rename', |
||||
'reference', |
||||
'redefine', |
||||
'prefix', |
||||
'once', |
||||
'old', |
||||
'obsolete', |
||||
'loop', |
||||
'local', |
||||
'like', |
||||
'is', |
||||
'inspect', |
||||
'infix', |
||||
'include', |
||||
'if', |
||||
'frozen', |
||||
'from', |
||||
'external', |
||||
'export', |
||||
'ensure', |
||||
'end', |
||||
'elseif', |
||||
'else', |
||||
'do', |
||||
'creation', |
||||
'create', |
||||
'check', |
||||
'alias', |
||||
'agent', |
||||
'separate', |
||||
'invariant', |
||||
'inherit', |
||||
'indexing', |
||||
'feature', |
||||
'expanded', |
||||
'deferred', |
||||
'class', |
||||
'Void', |
||||
'True', |
||||
'Result', |
||||
'Precursor', |
||||
'False', |
||||
'Current', |
||||
'create', |
||||
'attached', |
||||
'detachable', |
||||
'as', |
||||
'and', |
||||
'implies', |
||||
'not', |
||||
'or' |
||||
]); |
||||
var operators = wordObj([":=", "and then","and", "or","<<",">>"]); |
||||
|
||||
function chain(newtok, stream, state) { |
||||
state.tokenize.push(newtok); |
||||
return newtok(stream, state); |
||||
} |
||||
|
||||
function tokenBase(stream, state) { |
||||
if (stream.eatSpace()) return null; |
||||
var ch = stream.next(); |
||||
if (ch == '"'||ch == "'") { |
||||
return chain(readQuoted(ch, "string"), stream, state); |
||||
} else if (ch == "-"&&stream.eat("-")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} else if (ch == ":"&&stream.eat("=")) { |
||||
return "operator"; |
||||
} else if (/[0-9]/.test(ch)) { |
||||
stream.eatWhile(/[xXbBCc0-9\.]/); |
||||
stream.eat(/[\?\!]/); |
||||
return "ident"; |
||||
} else if (/[a-zA-Z_0-9]/.test(ch)) { |
||||
stream.eatWhile(/[a-zA-Z_0-9]/); |
||||
stream.eat(/[\?\!]/); |
||||
return "ident"; |
||||
} else if (/[=+\-\/*^%<>~]/.test(ch)) { |
||||
stream.eatWhile(/[=+\-\/*^%<>~]/); |
||||
return "operator"; |
||||
} else { |
||||
return null; |
||||
} |
||||
} |
||||
|
||||
function readQuoted(quote, style, unescaped) { |
||||
return function(stream, state) { |
||||
var escaped = false, ch; |
||||
while ((ch = stream.next()) != null) { |
||||
if (ch == quote && (unescaped || !escaped)) { |
||||
state.tokenize.pop(); |
||||
break; |
||||
} |
||||
escaped = !escaped && ch == "%"; |
||||
} |
||||
return style; |
||||
}; |
||||
} |
||||
|
||||
return { |
||||
startState: function() { |
||||
return {tokenize: [tokenBase]}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
var style = state.tokenize[state.tokenize.length-1](stream, state); |
||||
if (style == "ident") { |
||||
var word = stream.current(); |
||||
style = keywords.propertyIsEnumerable(stream.current()) ? "keyword" |
||||
: operators.propertyIsEnumerable(stream.current()) ? "operator" |
||||
: /^[A-Z][A-Z_0-9]*$/g.test(word) ? "tag" |
||||
: /^0[bB][0-1]+$/g.test(word) ? "number" |
||||
: /^0[cC][0-7]+$/g.test(word) ? "number" |
||||
: /^0[xX][a-fA-F0-9]+$/g.test(word) ? "number" |
||||
: /^([0-9]+\.[0-9]*)|([0-9]*\.[0-9]+)$/g.test(word) ? "number" |
||||
: /^[0-9]+$/g.test(word) ? "number" |
||||
: "variable"; |
||||
} |
||||
return style; |
||||
}, |
||||
lineComment: "--" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-eiffel", "eiffel"); |
||||
|
||||
}); |
@ -0,0 +1,429 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Eiffel mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/neat.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="eiffel.js"></script> |
||||
<style> |
||||
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;} |
||||
.cm-s-default span.cm-arrow { color: red; } |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Eiffel</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Eiffel mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
note |
||||
description: "[ |
||||
Project-wide universal properties. |
||||
This class is an ancestor to all developer-written classes. |
||||
ANY may be customized for individual projects or teams. |
||||
]" |
||||
|
||||
library: "Free implementation of ELKS library" |
||||
status: "See notice at end of class." |
||||
legal: "See notice at end of class." |
||||
date: "$Date: 2013-01-25 11:49:00 -0800 (Fri, 25 Jan 2013) $" |
||||
revision: "$Revision: 712 $" |
||||
|
||||
class |
||||
ANY |
||||
|
||||
feature -- Customization |
||||
|
||||
feature -- Access |
||||
|
||||
generator: STRING |
||||
-- Name of current object's generating class |
||||
-- (base class of the type of which it is a direct instance) |
||||
external |
||||
"built_in" |
||||
ensure |
||||
generator_not_void: Result /= Void |
||||
generator_not_empty: not Result.is_empty |
||||
end |
||||
|
||||
generating_type: TYPE [detachable like Current] |
||||
-- Type of current object |
||||
-- (type of which it is a direct instance) |
||||
do |
||||
Result := {detachable like Current} |
||||
ensure |
||||
generating_type_not_void: Result /= Void |
||||
end |
||||
|
||||
feature -- Status report |
||||
|
||||
conforms_to (other: ANY): BOOLEAN |
||||
-- Does type of current object conform to type |
||||
-- of `other' (as per Eiffel: The Language, chapter 13)? |
||||
require |
||||
other_not_void: other /= Void |
||||
external |
||||
"built_in" |
||||
end |
||||
|
||||
same_type (other: ANY): BOOLEAN |
||||
-- Is type of current object identical to type of `other'? |
||||
require |
||||
other_not_void: other /= Void |
||||
external |
||||
"built_in" |
||||
ensure |
||||
definition: Result = (conforms_to (other) and |
||||
other.conforms_to (Current)) |
||||
end |
||||
|
||||
feature -- Comparison |
||||
|
||||
is_equal (other: like Current): BOOLEAN |
||||
-- Is `other' attached to an object considered |
||||
-- equal to current object? |
||||
require |
||||
other_not_void: other /= Void |
||||
external |
||||
"built_in" |
||||
ensure |
||||
symmetric: Result implies other ~ Current |
||||
consistent: standard_is_equal (other) implies Result |
||||
end |
||||
|
||||
frozen standard_is_equal (other: like Current): BOOLEAN |
||||
-- Is `other' attached to an object of the same type |
||||
-- as current object, and field-by-field identical to it? |
||||
require |
||||
other_not_void: other /= Void |
||||
external |
||||
"built_in" |
||||
ensure |
||||
same_type: Result implies same_type (other) |
||||
symmetric: Result implies other.standard_is_equal (Current) |
||||
end |
||||
|
||||
frozen equal (a: detachable ANY; b: like a): BOOLEAN |
||||
-- Are `a' and `b' either both void or attached |
||||
-- to objects considered equal? |
||||
do |
||||
if a = Void then |
||||
Result := b = Void |
||||
else |
||||
Result := b /= Void and then |
||||
a.is_equal (b) |
||||
end |
||||
ensure |
||||
definition: Result = (a = Void and b = Void) or else |
||||
((a /= Void and b /= Void) and then |
||||
a.is_equal (b)) |
||||
end |
||||
|
||||
frozen standard_equal (a: detachable ANY; b: like a): BOOLEAN |
||||
-- Are `a' and `b' either both void or attached to |
||||
-- field-by-field identical objects of the same type? |
||||
-- Always uses default object comparison criterion. |
||||
do |
||||
if a = Void then |
||||
Result := b = Void |
||||
else |
||||
Result := b /= Void and then |
||||
a.standard_is_equal (b) |
||||
end |
||||
ensure |
||||
definition: Result = (a = Void and b = Void) or else |
||||
((a /= Void and b /= Void) and then |
||||
a.standard_is_equal (b)) |
||||
end |
||||
|
||||
frozen is_deep_equal (other: like Current): BOOLEAN |
||||
-- Are `Current' and `other' attached to isomorphic object structures? |
||||
require |
||||
other_not_void: other /= Void |
||||
external |
||||
"built_in" |
||||
ensure |
||||
shallow_implies_deep: standard_is_equal (other) implies Result |
||||
same_type: Result implies same_type (other) |
||||
symmetric: Result implies other.is_deep_equal (Current) |
||||
end |
||||
|
||||
frozen deep_equal (a: detachable ANY; b: like a): BOOLEAN |
||||
-- Are `a' and `b' either both void |
||||
-- or attached to isomorphic object structures? |
||||
do |
||||
if a = Void then |
||||
Result := b = Void |
||||
else |
||||
Result := b /= Void and then a.is_deep_equal (b) |
||||
end |
||||
ensure |
||||
shallow_implies_deep: standard_equal (a, b) implies Result |
||||
both_or_none_void: (a = Void) implies (Result = (b = Void)) |
||||
same_type: (Result and (a /= Void)) implies (b /= Void and then a.same_type (b)) |
||||
symmetric: Result implies deep_equal (b, a) |
||||
end |
||||
|
||||
feature -- Duplication |
||||
|
||||
frozen twin: like Current |
||||
-- New object equal to `Current' |
||||
-- `twin' calls `copy'; to change copying/twinning semantics, redefine `copy'. |
||||
external |
||||
"built_in" |
||||
ensure |
||||
twin_not_void: Result /= Void |
||||
is_equal: Result ~ Current |
||||
end |
||||
|
||||
copy (other: like Current) |
||||
-- Update current object using fields of object attached |
||||
-- to `other', so as to yield equal objects. |
||||
require |
||||
other_not_void: other /= Void |
||||
type_identity: same_type (other) |
||||
external |
||||
"built_in" |
||||
ensure |
||||
is_equal: Current ~ other |
||||
end |
||||
|
||||
frozen standard_copy (other: like Current) |
||||
-- Copy every field of `other' onto corresponding field |
||||
-- of current object. |
||||
require |
||||
other_not_void: other /= Void |
||||
type_identity: same_type (other) |
||||
external |
||||
"built_in" |
||||
ensure |
||||
is_standard_equal: standard_is_equal (other) |
||||
end |
||||
|
||||
frozen clone (other: detachable ANY): like other |
||||
-- Void if `other' is void; otherwise new object |
||||
-- equal to `other' |
||||
-- |
||||
-- For non-void `other', `clone' calls `copy'; |
||||
-- to change copying/cloning semantics, redefine `copy'. |
||||
obsolete |
||||
"Use `twin' instead." |
||||
do |
||||
if other /= Void then |
||||
Result := other.twin |
||||
end |
||||
ensure |
||||
equal: Result ~ other |
||||
end |
||||
|
||||
frozen standard_clone (other: detachable ANY): like other |
||||
-- Void if `other' is void; otherwise new object |
||||
-- field-by-field identical to `other'. |
||||
-- Always uses default copying semantics. |
||||
obsolete |
||||
"Use `standard_twin' instead." |
||||
do |
||||
if other /= Void then |
||||
Result := other.standard_twin |
||||
end |
||||
ensure |
||||
equal: standard_equal (Result, other) |
||||
end |
||||
|
||||
frozen standard_twin: like Current |
||||
-- New object field-by-field identical to `other'. |
||||
-- Always uses default copying semantics. |
||||
external |
||||
"built_in" |
||||
ensure |
||||
standard_twin_not_void: Result /= Void |
||||
equal: standard_equal (Result, Current) |
||||
end |
||||
|
||||
frozen deep_twin: like Current |
||||
-- New object structure recursively duplicated from Current. |
||||
external |
||||
"built_in" |
||||
ensure |
||||
deep_twin_not_void: Result /= Void |
||||
deep_equal: deep_equal (Current, Result) |
||||
end |
||||
|
||||
frozen deep_clone (other: detachable ANY): like other |
||||
-- Void if `other' is void: otherwise, new object structure |
||||
-- recursively duplicated from the one attached to `other' |
||||
obsolete |
||||
"Use `deep_twin' instead." |
||||
do |
||||
if other /= Void then |
||||
Result := other.deep_twin |
||||
end |
||||
ensure |
||||
deep_equal: deep_equal (other, Result) |
||||
end |
||||
|
||||
frozen deep_copy (other: like Current) |
||||
-- Effect equivalent to that of: |
||||
-- `copy' (`other' . `deep_twin') |
||||
require |
||||
other_not_void: other /= Void |
||||
do |
||||
copy (other.deep_twin) |
||||
ensure |
||||
deep_equal: deep_equal (Current, other) |
||||
end |
||||
|
||||
feature {NONE} -- Retrieval |
||||
|
||||
frozen internal_correct_mismatch |
||||
-- Called from runtime to perform a proper dynamic dispatch on `correct_mismatch' |
||||
-- from MISMATCH_CORRECTOR. |
||||
local |
||||
l_msg: STRING |
||||
l_exc: EXCEPTIONS |
||||
do |
||||
if attached {MISMATCH_CORRECTOR} Current as l_corrector then |
||||
l_corrector.correct_mismatch |
||||
else |
||||
create l_msg.make_from_string ("Mismatch: ") |
||||
create l_exc |
||||
l_msg.append (generating_type.name) |
||||
l_exc.raise_retrieval_exception (l_msg) |
||||
end |
||||
end |
||||
|
||||
feature -- Output |
||||
|
||||
io: STD_FILES |
||||
-- Handle to standard file setup |
||||
once |
||||
create Result |
||||
Result.set_output_default |
||||
ensure |
||||
io_not_void: Result /= Void |
||||
end |
||||
|
||||
out: STRING |
||||
-- New string containing terse printable representation |
||||
-- of current object |
||||
do |
||||
Result := tagged_out |
||||
ensure |
||||
out_not_void: Result /= Void |
||||
end |
||||
|
||||
frozen tagged_out: STRING |
||||
-- New string containing terse printable representation |
||||
-- of current object |
||||
external |
||||
"built_in" |
||||
ensure |
||||
tagged_out_not_void: Result /= Void |
||||
end |
||||
|
||||
print (o: detachable ANY) |
||||
-- Write terse external representation of `o' |
||||
-- on standard output. |
||||
do |
||||
if o /= Void then |
||||
io.put_string (o.out) |
||||
end |
||||
end |
||||
|
||||
feature -- Platform |
||||
|
||||
Operating_environment: OPERATING_ENVIRONMENT |
||||
-- Objects available from the operating system |
||||
once |
||||
create Result |
||||
ensure |
||||
operating_environment_not_void: Result /= Void |
||||
end |
||||
|
||||
feature {NONE} -- Initialization |
||||
|
||||
default_create |
||||
-- Process instances of classes with no creation clause. |
||||
-- (Default: do nothing.) |
||||
do |
||||
end |
||||
|
||||
feature -- Basic operations |
||||
|
||||
default_rescue |
||||
-- Process exception for routines with no Rescue clause. |
||||
-- (Default: do nothing.) |
||||
do |
||||
end |
||||
|
||||
frozen do_nothing |
||||
-- Execute a null action. |
||||
do |
||||
end |
||||
|
||||
frozen default: detachable like Current |
||||
-- Default value of object's type |
||||
do |
||||
end |
||||
|
||||
frozen default_pointer: POINTER |
||||
-- Default value of type `POINTER' |
||||
-- (Avoid the need to write `p'.`default' for |
||||
-- some `p' of type `POINTER'.) |
||||
do |
||||
ensure |
||||
-- Result = Result.default |
||||
end |
||||
|
||||
frozen as_attached: attached like Current |
||||
-- Attached version of Current |
||||
-- (Can be used during transitional period to convert |
||||
-- non-void-safe classes to void-safe ones.) |
||||
do |
||||
Result := Current |
||||
end |
||||
|
||||
invariant |
||||
reflexive_equality: standard_is_equal (Current) |
||||
reflexive_conformance: conforms_to (Current) |
||||
|
||||
note |
||||
copyright: "Copyright (c) 1984-2012, Eiffel Software and others" |
||||
license: "Eiffel Forum License v2 (see http://www.eiffel.com/licensing/forum.txt)" |
||||
source: "[ |
||||
Eiffel Software |
||||
5949 Hollister Ave., Goleta, CA 93117 USA |
||||
Telephone 805-685-1006, Fax 805-685-6869 |
||||
Website http://www.eiffel.com |
||||
Customer support http://support.eiffel.com |
||||
]" |
||||
|
||||
end |
||||
|
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: "text/x-eiffel", |
||||
indentUnit: 4, |
||||
lineNumbers: true, |
||||
theme: "neat" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-eiffel</code>.</p> |
||||
|
||||
<p> Created by <a href="https://github.com/ynh">YNH</a>.</p> |
||||
</article> |
@ -0,0 +1,205 @@ |
||||
// 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("elm", function() { |
||||
|
||||
function switchState(source, setState, f) { |
||||
setState(f); |
||||
return f(source, setState); |
||||
} |
||||
|
||||
// These should all be Unicode extended, as per the Haskell 2010 report
|
||||
var smallRE = /[a-z_]/; |
||||
var largeRE = /[A-Z]/; |
||||
var digitRE = /[0-9]/; |
||||
var hexitRE = /[0-9A-Fa-f]/; |
||||
var octitRE = /[0-7]/; |
||||
var idRE = /[a-z_A-Z0-9\']/; |
||||
var symbolRE = /[-!#$%&*+.\/<=>?@\\^|~:\u03BB\u2192]/; |
||||
var specialRE = /[(),;[\]`{}]/; |
||||
var whiteCharRE = /[ \t\v\f]/; // newlines are handled in tokenizer
|
||||
|
||||
function normal() { |
||||
return function (source, setState) { |
||||
if (source.eatWhile(whiteCharRE)) { |
||||
return null; |
||||
} |
||||
|
||||
var ch = source.next(); |
||||
if (specialRE.test(ch)) { |
||||
if (ch == '{' && source.eat('-')) { |
||||
var t = "comment"; |
||||
if (source.eat('#')) t = "meta"; |
||||
return switchState(source, setState, ncomment(t, 1)); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
if (ch == '\'') { |
||||
if (source.eat('\\')) |
||||
source.next(); // should handle other escapes here
|
||||
else |
||||
source.next(); |
||||
|
||||
if (source.eat('\'')) |
||||
return "string"; |
||||
return "error"; |
||||
} |
||||
|
||||
if (ch == '"') { |
||||
return switchState(source, setState, stringLiteral); |
||||
} |
||||
|
||||
if (largeRE.test(ch)) { |
||||
source.eatWhile(idRE); |
||||
if (source.eat('.')) |
||||
return "qualifier"; |
||||
return "variable-2"; |
||||
} |
||||
|
||||
if (smallRE.test(ch)) { |
||||
var isDef = source.pos === 1; |
||||
source.eatWhile(idRE); |
||||
return isDef ? "variable-3" : "variable"; |
||||
} |
||||
|
||||
if (digitRE.test(ch)) { |
||||
if (ch == '0') { |
||||
if (source.eat(/[xX]/)) { |
||||
source.eatWhile(hexitRE); // should require at least 1
|
||||
return "integer"; |
||||
} |
||||
if (source.eat(/[oO]/)) { |
||||
source.eatWhile(octitRE); // should require at least 1
|
||||
return "number"; |
||||
} |
||||
} |
||||
source.eatWhile(digitRE); |
||||
var t = "number"; |
||||
if (source.eat('.')) { |
||||
t = "number"; |
||||
source.eatWhile(digitRE); // should require at least 1
|
||||
} |
||||
if (source.eat(/[eE]/)) { |
||||
t = "number"; |
||||
source.eat(/[-+]/); |
||||
source.eatWhile(digitRE); // should require at least 1
|
||||
} |
||||
return t; |
||||
} |
||||
|
||||
if (symbolRE.test(ch)) { |
||||
if (ch == '-' && source.eat(/-/)) { |
||||
source.eatWhile(/-/); |
||||
if (!source.eat(symbolRE)) { |
||||
source.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
source.eatWhile(symbolRE); |
||||
return "builtin"; |
||||
} |
||||
|
||||
return "error"; |
||||
} |
||||
} |
||||
|
||||
function ncomment(type, nest) { |
||||
if (nest == 0) { |
||||
return normal(); |
||||
} |
||||
return function(source, setState) { |
||||
var currNest = nest; |
||||
while (!source.eol()) { |
||||
var ch = source.next(); |
||||
if (ch == '{' && source.eat('-')) { |
||||
++currNest; |
||||
} else if (ch == '-' && source.eat('}')) { |
||||
--currNest; |
||||
if (currNest == 0) { |
||||
setState(normal()); |
||||
return type; |
||||
} |
||||
} |
||||
} |
||||
setState(ncomment(type, currNest)); |
||||
return type; |
||||
} |
||||
} |
||||
|
||||
function stringLiteral(source, setState) { |
||||
while (!source.eol()) { |
||||
var ch = source.next(); |
||||
if (ch == '"') { |
||||
setState(normal()); |
||||
return "string"; |
||||
} |
||||
if (ch == '\\') { |
||||
if (source.eol() || source.eat(whiteCharRE)) { |
||||
setState(stringGap); |
||||
return "string"; |
||||
} |
||||
if (!source.eat('&')) source.next(); // should handle other escapes here
|
||||
} |
||||
} |
||||
setState(normal()); |
||||
return "error"; |
||||
} |
||||
|
||||
function stringGap(source, setState) { |
||||
if (source.eat('\\')) { |
||||
return switchState(source, setState, stringLiteral); |
||||
} |
||||
source.next(); |
||||
setState(normal()); |
||||
return "error"; |
||||
} |
||||
|
||||
|
||||
var wellKnownWords = (function() { |
||||
var wkw = {}; |
||||
|
||||
var keywords = [ |
||||
"case", "of", "as", |
||||
"if", "then", "else", |
||||
"let", "in", |
||||
"infix", "infixl", "infixr", |
||||
"type", "alias", |
||||
"input", "output", "foreign", "loopback", |
||||
"module", "where", "import", "exposing", |
||||
"_", "..", "|", ":", "=", "\\", "\"", "->", "<-" |
||||
]; |
||||
|
||||
for (var i = keywords.length; i--;) |
||||
wkw[keywords[i]] = "keyword"; |
||||
|
||||
return wkw; |
||||
})(); |
||||
|
||||
|
||||
|
||||
return { |
||||
startState: function () { return { f: normal() }; }, |
||||
copyState: function (s) { return { f: s.f }; }, |
||||
|
||||
token: function(stream, state) { |
||||
var t = state.f(stream, function(s) { state.f = s; }); |
||||
var w = stream.current(); |
||||
return (wellKnownWords.hasOwnProperty(w)) ? wellKnownWords[w] : t; |
||||
} |
||||
}; |
||||
|
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-elm", "elm"); |
||||
}); |
@ -0,0 +1,61 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Elm mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="elm.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Elm</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Elm mode</h2> |
||||
|
||||
<div><textarea id="code" name="code"> |
||||
import Color exposing (..) |
||||
import Graphics.Collage exposing (..) |
||||
import Graphics.Element exposing (..) |
||||
import Time exposing (..) |
||||
|
||||
main = |
||||
Signal.map clock (every second) |
||||
|
||||
clock t = |
||||
collage 400 400 |
||||
[ filled lightGrey (ngon 12 110) |
||||
, outlined (solid grey) (ngon 12 110) |
||||
, hand orange 100 t |
||||
, hand charcoal 100 (t/60) |
||||
, hand charcoal 60 (t/720) |
||||
] |
||||
|
||||
hand clr len time = |
||||
let angle = degrees (90 - 6 * inSeconds time) |
||||
in |
||||
segment (0,0) (fromPolar (len,angle)) |
||||
|> traced (solid clr) |
||||
</textarea></div> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "text/x-elm" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-elm</code>.</p> |
||||
</article> |
@ -0,0 +1,618 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
/*jshint unused:true, eqnull:true, curly:true, bitwise:true */ |
||||
/*jshint undef:true, latedef:true, trailing:true */ |
||||
/*global CodeMirror:true */ |
||||
|
||||
// erlang mode.
|
||||
// tokenizer -> token types -> CodeMirror styles
|
||||
// tokenizer maintains a parse stack
|
||||
// indenter uses the parse stack
|
||||
|
||||
// TODO indenter:
|
||||
// bit syntax
|
||||
// old guard/bif/conversion clashes (e.g. "float/1")
|
||||
// type/spec/opaque
|
||||
|
||||
(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.defineMIME("text/x-erlang", "erlang"); |
||||
|
||||
CodeMirror.defineMode("erlang", function(cmCfg) { |
||||
"use strict"; |
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// constants
|
||||
|
||||
var typeWords = [ |
||||
"-type", "-spec", "-export_type", "-opaque"]; |
||||
|
||||
var keywordWords = [ |
||||
"after","begin","catch","case","cond","end","fun","if", |
||||
"let","of","query","receive","try","when"]; |
||||
|
||||
var separatorRE = /[\->,;]/; |
||||
var separatorWords = [ |
||||
"->",";",","]; |
||||
|
||||
var operatorAtomWords = [ |
||||
"and","andalso","band","bnot","bor","bsl","bsr","bxor", |
||||
"div","not","or","orelse","rem","xor"]; |
||||
|
||||
var operatorSymbolRE = /[\+\-\*\/<>=\|:!]/; |
||||
var operatorSymbolWords = [ |
||||
"=","+","-","*","/",">",">=","<","=<","=:=","==","=/=","/=","||","<-","!"]; |
||||
|
||||
var openParenRE = /[<\(\[\{]/; |
||||
var openParenWords = [ |
||||
"<<","(","[","{"]; |
||||
|
||||
var closeParenRE = /[>\)\]\}]/; |
||||
var closeParenWords = [ |
||||
"}","]",")",">>"]; |
||||
|
||||
var guardWords = [ |
||||
"is_atom","is_binary","is_bitstring","is_boolean","is_float", |
||||
"is_function","is_integer","is_list","is_number","is_pid", |
||||
"is_port","is_record","is_reference","is_tuple", |
||||
"atom","binary","bitstring","boolean","function","integer","list", |
||||
"number","pid","port","record","reference","tuple"]; |
||||
|
||||
var bifWords = [ |
||||
"abs","adler32","adler32_combine","alive","apply","atom_to_binary", |
||||
"atom_to_list","binary_to_atom","binary_to_existing_atom", |
||||
"binary_to_list","binary_to_term","bit_size","bitstring_to_list", |
||||
"byte_size","check_process_code","contact_binary","crc32", |
||||
"crc32_combine","date","decode_packet","delete_module", |
||||
"disconnect_node","element","erase","exit","float","float_to_list", |
||||
"garbage_collect","get","get_keys","group_leader","halt","hd", |
||||
"integer_to_list","internal_bif","iolist_size","iolist_to_binary", |
||||
"is_alive","is_atom","is_binary","is_bitstring","is_boolean", |
||||
"is_float","is_function","is_integer","is_list","is_number","is_pid", |
||||
"is_port","is_process_alive","is_record","is_reference","is_tuple", |
||||
"length","link","list_to_atom","list_to_binary","list_to_bitstring", |
||||
"list_to_existing_atom","list_to_float","list_to_integer", |
||||
"list_to_pid","list_to_tuple","load_module","make_ref","module_loaded", |
||||
"monitor_node","node","node_link","node_unlink","nodes","notalive", |
||||
"now","open_port","pid_to_list","port_close","port_command", |
||||
"port_connect","port_control","pre_loaded","process_flag", |
||||
"process_info","processes","purge_module","put","register", |
||||
"registered","round","self","setelement","size","spawn","spawn_link", |
||||
"spawn_monitor","spawn_opt","split_binary","statistics", |
||||
"term_to_binary","time","throw","tl","trunc","tuple_size", |
||||
"tuple_to_list","unlink","unregister","whereis"]; |
||||
|
||||
// upper case: [A-Z] [Ø-Þ] [À-Ö]
|
||||
// lower case: [a-z] [ß-ö] [ø-ÿ]
|
||||
var anumRE = /[\w@Ø-ÞÀ-Öß-öø-ÿ]/; |
||||
var escapesRE = |
||||
/[0-7]{1,3}|[bdefnrstv\\"']|\^[a-zA-Z]|x[0-9a-zA-Z]{2}|x{[0-9a-zA-Z]+}/; |
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// tokenizer
|
||||
|
||||
function tokenizer(stream,state) { |
||||
// in multi-line string
|
||||
if (state.in_string) { |
||||
state.in_string = (!doubleQuote(stream)); |
||||
return rval(state,stream,"string"); |
||||
} |
||||
|
||||
// in multi-line atom
|
||||
if (state.in_atom) { |
||||
state.in_atom = (!singleQuote(stream)); |
||||
return rval(state,stream,"atom"); |
||||
} |
||||
|
||||
// whitespace
|
||||
if (stream.eatSpace()) { |
||||
return rval(state,stream,"whitespace"); |
||||
} |
||||
|
||||
// attributes and type specs
|
||||
if (!peekToken(state) && |
||||
stream.match(/-\s*[a-zß-öø-ÿ][\wØ-ÞÀ-Öß-öø-ÿ]*/)) { |
||||
if (is_member(stream.current(),typeWords)) { |
||||
return rval(state,stream,"type"); |
||||
}else{ |
||||
return rval(state,stream,"attribute"); |
||||
} |
||||
} |
||||
|
||||
var ch = stream.next(); |
||||
|
||||
// comment
|
||||
if (ch == '%') { |
||||
stream.skipToEnd(); |
||||
return rval(state,stream,"comment"); |
||||
} |
||||
|
||||
// colon
|
||||
if (ch == ":") { |
||||
return rval(state,stream,"colon"); |
||||
} |
||||
|
||||
// macro
|
||||
if (ch == '?') { |
||||
stream.eatSpace(); |
||||
stream.eatWhile(anumRE); |
||||
return rval(state,stream,"macro"); |
||||
} |
||||
|
||||
// record
|
||||
if (ch == "#") { |
||||
stream.eatSpace(); |
||||
stream.eatWhile(anumRE); |
||||
return rval(state,stream,"record"); |
||||
} |
||||
|
||||
// dollar escape
|
||||
if (ch == "$") { |
||||
if (stream.next() == "\\" && !stream.match(escapesRE)) { |
||||
return rval(state,stream,"error"); |
||||
} |
||||
return rval(state,stream,"number"); |
||||
} |
||||
|
||||
// dot
|
||||
if (ch == ".") { |
||||
return rval(state,stream,"dot"); |
||||
} |
||||
|
||||
// quoted atom
|
||||
if (ch == '\'') { |
||||
if (!(state.in_atom = (!singleQuote(stream)))) { |
||||
if (stream.match(/\s*\/\s*[0-9]/,false)) { |
||||
stream.match(/\s*\/\s*[0-9]/,true); |
||||
return rval(state,stream,"fun"); // 'f'/0 style fun
|
||||
} |
||||
if (stream.match(/\s*\(/,false) || stream.match(/\s*:/,false)) { |
||||
return rval(state,stream,"function"); |
||||
} |
||||
} |
||||
return rval(state,stream,"atom"); |
||||
} |
||||
|
||||
// string
|
||||
if (ch == '"') { |
||||
state.in_string = (!doubleQuote(stream)); |
||||
return rval(state,stream,"string"); |
||||
} |
||||
|
||||
// variable
|
||||
if (/[A-Z_Ø-ÞÀ-Ö]/.test(ch)) { |
||||
stream.eatWhile(anumRE); |
||||
return rval(state,stream,"variable"); |
||||
} |
||||
|
||||
// atom/keyword/BIF/function
|
||||
if (/[a-z_ß-öø-ÿ]/.test(ch)) { |
||||
stream.eatWhile(anumRE); |
||||
|
||||
if (stream.match(/\s*\/\s*[0-9]/,false)) { |
||||
stream.match(/\s*\/\s*[0-9]/,true); |
||||
return rval(state,stream,"fun"); // f/0 style fun
|
||||
} |
||||
|
||||
var w = stream.current(); |
||||
|
||||
if (is_member(w,keywordWords)) { |
||||
return rval(state,stream,"keyword"); |
||||
}else if (is_member(w,operatorAtomWords)) { |
||||
return rval(state,stream,"operator"); |
||||
}else if (stream.match(/\s*\(/,false)) { |
||||
// 'put' and 'erlang:put' are bifs, 'foo:put' is not
|
||||
if (is_member(w,bifWords) && |
||||
((peekToken(state).token != ":") || |
||||
(peekToken(state,2).token == "erlang"))) { |
||||
return rval(state,stream,"builtin"); |
||||
}else if (is_member(w,guardWords)) { |
||||
return rval(state,stream,"guard"); |
||||
}else{ |
||||
return rval(state,stream,"function"); |
||||
} |
||||
}else if (lookahead(stream) == ":") { |
||||
if (w == "erlang") { |
||||
return rval(state,stream,"builtin"); |
||||
} else { |
||||
return rval(state,stream,"function"); |
||||
} |
||||
}else if (is_member(w,["true","false"])) { |
||||
return rval(state,stream,"boolean"); |
||||
}else{ |
||||
return rval(state,stream,"atom"); |
||||
} |
||||
} |
||||
|
||||
// number
|
||||
var digitRE = /[0-9]/; |
||||
var radixRE = /[0-9a-zA-Z]/; // 36#zZ style int
|
||||
if (digitRE.test(ch)) { |
||||
stream.eatWhile(digitRE); |
||||
if (stream.eat('#')) { // 36#aZ style integer
|
||||
if (!stream.eatWhile(radixRE)) { |
||||
stream.backUp(1); //"36#" - syntax error
|
||||
} |
||||
} else if (stream.eat('.')) { // float
|
||||
if (!stream.eatWhile(digitRE)) { |
||||
stream.backUp(1); // "3." - probably end of function
|
||||
} else { |
||||
if (stream.eat(/[eE]/)) { // float with exponent
|
||||
if (stream.eat(/[-+]/)) { |
||||
if (!stream.eatWhile(digitRE)) { |
||||
stream.backUp(2); // "2e-" - syntax error
|
||||
} |
||||
} else { |
||||
if (!stream.eatWhile(digitRE)) { |
||||
stream.backUp(1); // "2e" - syntax error
|
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
return rval(state,stream,"number"); // normal integer
|
||||
} |
||||
|
||||
// open parens
|
||||
if (nongreedy(stream,openParenRE,openParenWords)) { |
||||
return rval(state,stream,"open_paren"); |
||||
} |
||||
|
||||
// close parens
|
||||
if (nongreedy(stream,closeParenRE,closeParenWords)) { |
||||
return rval(state,stream,"close_paren"); |
||||
} |
||||
|
||||
// separators
|
||||
if (greedy(stream,separatorRE,separatorWords)) { |
||||
return rval(state,stream,"separator"); |
||||
} |
||||
|
||||
// operators
|
||||
if (greedy(stream,operatorSymbolRE,operatorSymbolWords)) { |
||||
return rval(state,stream,"operator"); |
||||
} |
||||
|
||||
return rval(state,stream,null); |
||||
} |
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// utilities
|
||||
function nongreedy(stream,re,words) { |
||||
if (stream.current().length == 1 && re.test(stream.current())) { |
||||
stream.backUp(1); |
||||
while (re.test(stream.peek())) { |
||||
stream.next(); |
||||
if (is_member(stream.current(),words)) { |
||||
return true; |
||||
} |
||||
} |
||||
stream.backUp(stream.current().length-1); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
function greedy(stream,re,words) { |
||||
if (stream.current().length == 1 && re.test(stream.current())) { |
||||
while (re.test(stream.peek())) { |
||||
stream.next(); |
||||
} |
||||
while (0 < stream.current().length) { |
||||
if (is_member(stream.current(),words)) { |
||||
return true; |
||||
}else{ |
||||
stream.backUp(1); |
||||
} |
||||
} |
||||
stream.next(); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
function doubleQuote(stream) { |
||||
return quote(stream, '"', '\\'); |
||||
} |
||||
|
||||
function singleQuote(stream) { |
||||
return quote(stream,'\'','\\'); |
||||
} |
||||
|
||||
function quote(stream,quoteChar,escapeChar) { |
||||
while (!stream.eol()) { |
||||
var ch = stream.next(); |
||||
if (ch == quoteChar) { |
||||
return true; |
||||
}else if (ch == escapeChar) { |
||||
stream.next(); |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
function lookahead(stream) { |
||||
var m = stream.match(/([\n\s]+|%[^\n]*\n)*(.)/,false); |
||||
return m ? m.pop() : ""; |
||||
} |
||||
|
||||
function is_member(element,list) { |
||||
return (-1 < list.indexOf(element)); |
||||
} |
||||
|
||||
function rval(state,stream,type) { |
||||
|
||||
// parse stack
|
||||
pushToken(state,realToken(type,stream)); |
||||
|
||||
// map erlang token type to CodeMirror style class
|
||||
// erlang -> CodeMirror tag
|
||||
switch (type) { |
||||
case "atom": return "atom"; |
||||
case "attribute": return "attribute"; |
||||
case "boolean": return "atom"; |
||||
case "builtin": return "builtin"; |
||||
case "close_paren": return null; |
||||
case "colon": return null; |
||||
case "comment": return "comment"; |
||||
case "dot": return null; |
||||
case "error": return "error"; |
||||
case "fun": return "meta"; |
||||
case "function": return "tag"; |
||||
case "guard": return "property"; |
||||
case "keyword": return "keyword"; |
||||
case "macro": return "variable-2"; |
||||
case "number": return "number"; |
||||
case "open_paren": return null; |
||||
case "operator": return "operator"; |
||||
case "record": return "bracket"; |
||||
case "separator": return null; |
||||
case "string": return "string"; |
||||
case "type": return "def"; |
||||
case "variable": return "variable"; |
||||
default: return null; |
||||
} |
||||
} |
||||
|
||||
function aToken(tok,col,ind,typ) { |
||||
return {token: tok, |
||||
column: col, |
||||
indent: ind, |
||||
type: typ}; |
||||
} |
||||
|
||||
function realToken(type,stream) { |
||||
return aToken(stream.current(), |
||||
stream.column(), |
||||
stream.indentation(), |
||||
type); |
||||
} |
||||
|
||||
function fakeToken(type) { |
||||
return aToken(type,0,0,type); |
||||
} |
||||
|
||||
function peekToken(state,depth) { |
||||
var len = state.tokenStack.length; |
||||
var dep = (depth ? depth : 1); |
||||
|
||||
if (len < dep) { |
||||
return false; |
||||
}else{ |
||||
return state.tokenStack[len-dep]; |
||||
} |
||||
} |
||||
|
||||
function pushToken(state,token) { |
||||
|
||||
if (!(token.type == "comment" || token.type == "whitespace")) { |
||||
state.tokenStack = maybe_drop_pre(state.tokenStack,token); |
||||
state.tokenStack = maybe_drop_post(state.tokenStack); |
||||
} |
||||
} |
||||
|
||||
function maybe_drop_pre(s,token) { |
||||
var last = s.length-1; |
||||
|
||||
if (0 < last && s[last].type === "record" && token.type === "dot") { |
||||
s.pop(); |
||||
}else if (0 < last && s[last].type === "group") { |
||||
s.pop(); |
||||
s.push(token); |
||||
}else{ |
||||
s.push(token); |
||||
} |
||||
return s; |
||||
} |
||||
|
||||
function maybe_drop_post(s) { |
||||
var last = s.length-1; |
||||
|
||||
if (s[last].type === "dot") { |
||||
return []; |
||||
} |
||||
if (s[last].type === "fun" && s[last-1].token === "fun") { |
||||
return s.slice(0,last-1); |
||||
} |
||||
switch (s[s.length-1].token) { |
||||
case "}": return d(s,{g:["{"]}); |
||||
case "]": return d(s,{i:["["]}); |
||||
case ")": return d(s,{i:["("]}); |
||||
case ">>": return d(s,{i:["<<"]}); |
||||
case "end": return d(s,{i:["begin","case","fun","if","receive","try"]}); |
||||
case ",": return d(s,{e:["begin","try","when","->", |
||||
",","(","[","{","<<"]}); |
||||
case "->": return d(s,{r:["when"], |
||||
m:["try","if","case","receive"]}); |
||||
case ";": return d(s,{E:["case","fun","if","receive","try","when"]}); |
||||
case "catch":return d(s,{e:["try"]}); |
||||
case "of": return d(s,{e:["case"]}); |
||||
case "after":return d(s,{e:["receive","try"]}); |
||||
default: return s; |
||||
} |
||||
} |
||||
|
||||
function d(stack,tt) { |
||||
// stack is a stack of Token objects.
|
||||
// tt is an object; {type:tokens}
|
||||
// type is a char, tokens is a list of token strings.
|
||||
// The function returns (possibly truncated) stack.
|
||||
// It will descend the stack, looking for a Token such that Token.token
|
||||
// is a member of tokens. If it does not find that, it will normally (but
|
||||
// see "E" below) return stack. If it does find a match, it will remove
|
||||
// all the Tokens between the top and the matched Token.
|
||||
// If type is "m", that is all it does.
|
||||
// If type is "i", it will also remove the matched Token and the top Token.
|
||||
// If type is "g", like "i", but add a fake "group" token at the top.
|
||||
// If type is "r", it will remove the matched Token, but not the top Token.
|
||||
// If type is "e", it will keep the matched Token but not the top Token.
|
||||
// If type is "E", it behaves as for type "e", except if there is no match,
|
||||
// in which case it will return an empty stack.
|
||||
|
||||
for (var type in tt) { |
||||
var len = stack.length-1; |
||||
var tokens = tt[type]; |
||||
for (var i = len-1; -1 < i ; i--) { |
||||
if (is_member(stack[i].token,tokens)) { |
||||
var ss = stack.slice(0,i); |
||||
switch (type) { |
||||
case "m": return ss.concat(stack[i]).concat(stack[len]); |
||||
case "r": return ss.concat(stack[len]); |
||||
case "i": return ss; |
||||
case "g": return ss.concat(fakeToken("group")); |
||||
case "E": return ss.concat(stack[i]); |
||||
case "e": return ss.concat(stack[i]); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
return (type == "E" ? [] : stack); |
||||
} |
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// indenter
|
||||
|
||||
function indenter(state,textAfter) { |
||||
var t; |
||||
var unit = cmCfg.indentUnit; |
||||
var wordAfter = wordafter(textAfter); |
||||
var currT = peekToken(state,1); |
||||
var prevT = peekToken(state,2); |
||||
|
||||
if (state.in_string || state.in_atom) { |
||||
return CodeMirror.Pass; |
||||
}else if (!prevT) { |
||||
return 0; |
||||
}else if (currT.token == "when") { |
||||
return currT.column+unit; |
||||
}else if (wordAfter === "when" && prevT.type === "function") { |
||||
return prevT.indent+unit; |
||||
}else if (wordAfter === "(" && currT.token === "fun") { |
||||
return currT.column+3; |
||||
}else if (wordAfter === "catch" && (t = getToken(state,["try"]))) { |
||||
return t.column; |
||||
}else if (is_member(wordAfter,["end","after","of"])) { |
||||
t = getToken(state,["begin","case","fun","if","receive","try"]); |
||||
return t ? t.column : CodeMirror.Pass; |
||||
}else if (is_member(wordAfter,closeParenWords)) { |
||||
t = getToken(state,openParenWords); |
||||
return t ? t.column : CodeMirror.Pass; |
||||
}else if (is_member(currT.token,[",","|","||"]) || |
||||
is_member(wordAfter,[",","|","||"])) { |
||||
t = postcommaToken(state); |
||||
return t ? t.column+t.token.length : unit; |
||||
}else if (currT.token == "->") { |
||||
if (is_member(prevT.token, ["receive","case","if","try"])) { |
||||
return prevT.column+unit+unit; |
||||
}else{ |
||||
return prevT.column+unit; |
||||
} |
||||
}else if (is_member(currT.token,openParenWords)) { |
||||
return currT.column+currT.token.length; |
||||
}else{ |
||||
t = defaultToken(state); |
||||
return truthy(t) ? t.column+unit : 0; |
||||
} |
||||
} |
||||
|
||||
function wordafter(str) { |
||||
var m = str.match(/,|[a-z]+|\}|\]|\)|>>|\|+|\(/); |
||||
|
||||
return truthy(m) && (m.index === 0) ? m[0] : ""; |
||||
} |
||||
|
||||
function postcommaToken(state) { |
||||
var objs = state.tokenStack.slice(0,-1); |
||||
var i = getTokenIndex(objs,"type",["open_paren"]); |
||||
|
||||
return truthy(objs[i]) ? objs[i] : false; |
||||
} |
||||
|
||||
function defaultToken(state) { |
||||
var objs = state.tokenStack; |
||||
var stop = getTokenIndex(objs,"type",["open_paren","separator","keyword"]); |
||||
var oper = getTokenIndex(objs,"type",["operator"]); |
||||
|
||||
if (truthy(stop) && truthy(oper) && stop < oper) { |
||||
return objs[stop+1]; |
||||
} else if (truthy(stop)) { |
||||
return objs[stop]; |
||||
} else { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
function getToken(state,tokens) { |
||||
var objs = state.tokenStack; |
||||
var i = getTokenIndex(objs,"token",tokens); |
||||
|
||||
return truthy(objs[i]) ? objs[i] : false; |
||||
} |
||||
|
||||
function getTokenIndex(objs,propname,propvals) { |
||||
|
||||
for (var i = objs.length-1; -1 < i ; i--) { |
||||
if (is_member(objs[i][propname],propvals)) { |
||||
return i; |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
function truthy(x) { |
||||
return (x !== false) && (x != null); |
||||
} |
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// this object defines the mode
|
||||
|
||||
return { |
||||
startState: |
||||
function() { |
||||
return {tokenStack: [], |
||||
in_string: false, |
||||
in_atom: false}; |
||||
}, |
||||
|
||||
token: |
||||
function(stream, state) { |
||||
return tokenizer(stream, state); |
||||
}, |
||||
|
||||
indent: |
||||
function(state, textAfter) { |
||||
return indenter(state,textAfter); |
||||
}, |
||||
|
||||
lineComment: "%" |
||||
}; |
||||
}); |
||||
|
||||
}); |
@ -0,0 +1,76 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Erlang mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/erlang-dark.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="erlang.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Erlang</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Erlang mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
%% -*- mode: erlang; erlang-indent-level: 2 -*- |
||||
%%% Created : 7 May 2012 by mats cronqvist <masse@klarna.com> |
||||
|
||||
%% @doc |
||||
%% Demonstrates how to print a record. |
||||
%% @end |
||||
|
||||
-module('ex'). |
||||
-author('mats cronqvist'). |
||||
-export([demo/0, |
||||
rec_info/1]). |
||||
|
||||
-record(demo,{a="One",b="Two",c="Three",d="Four"}). |
||||
|
||||
rec_info(demo) -> record_info(fields,demo). |
||||
|
||||
demo() -> expand_recs(?MODULE,#demo{a="A",b="BB"}). |
||||
|
||||
expand_recs(M,List) when is_list(List) -> |
||||
[expand_recs(M,L)||L<-List]; |
||||
expand_recs(M,Tup) when is_tuple(Tup) -> |
||||
case tuple_size(Tup) of |
||||
L when L < 1 -> Tup; |
||||
L -> |
||||
try |
||||
Fields = M:rec_info(element(1,Tup)), |
||||
L = length(Fields)+1, |
||||
lists:zip(Fields,expand_recs(M,tl(tuple_to_list(Tup)))) |
||||
catch |
||||
_:_ -> list_to_tuple(expand_recs(M,tuple_to_list(Tup))) |
||||
end |
||||
end; |
||||
expand_recs(_,Term) -> |
||||
Term. |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
extraKeys: {"Tab": "indentAuto"}, |
||||
theme: "erlang-dark" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-erlang</code>.</p> |
||||
</article> |
@ -0,0 +1,83 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
// Factor syntax highlight - simple mode
|
||||
//
|
||||
// by Dimage Sapelkin (https://github.com/kerabromsmu)
|
||||
|
||||
(function(mod) { |
||||
if (typeof exports == "object" && typeof module == "object") // CommonJS
|
||||
mod(require("../../lib/codemirror"), require("../../addon/mode/simple")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../../addon/mode/simple"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
CodeMirror.defineSimpleMode("factor", { |
||||
// The start state contains the rules that are intially used
|
||||
start: [ |
||||
// comments
|
||||
{regex: /#?!.*/, token: "comment"}, |
||||
// strings """, multiline --> state
|
||||
{regex: /"""/, token: "string", next: "string3"}, |
||||
{regex: /"/, token: "string", next: "string"}, |
||||
// numbers: dec, hex, unicode, bin, fractional, complex
|
||||
{regex: /(?:[+-]?)(?:0x[\d,a-f]+)|(?:0o[0-7]+)|(?:0b[0,1]+)|(?:\d+.?\d*)/, token: "number"}, |
||||
//{regex: /[+-]?/} //fractional
|
||||
// definition: defining word, defined word, etc
|
||||
{regex: /(\:)(\s+)(\S+)(\s+)(\()/, token: ["keyword", null, "def", null, "keyword"], next: "stack"}, |
||||
// vocabulary using --> state
|
||||
{regex: /USING\:/, token: "keyword", next: "vocabulary"}, |
||||
// vocabulary definition/use
|
||||
{regex: /(USE\:|IN\:)(\s+)(\S+)/, token: ["keyword", null, "variable-2"]}, |
||||
// <constructors>
|
||||
{regex: /<\S+>/, token: "builtin"}, |
||||
// "keywords", incl. ; t f . [ ] { } defining words
|
||||
{regex: /;|t|f|if|\.|\[|\]|\{|\}|MAIN:/, token: "keyword"}, |
||||
// any id (?)
|
||||
{regex: /\S+/, token: "variable"}, |
||||
|
||||
{ |
||||
regex: /./, |
||||
token: null |
||||
} |
||||
], |
||||
vocabulary: [ |
||||
{regex: /;/, token: "keyword", next: "start"}, |
||||
{regex: /\S+/, token: "variable-2"}, |
||||
{ |
||||
regex: /./, |
||||
token: null |
||||
} |
||||
], |
||||
string: [ |
||||
{regex: /(?:[^\\]|\\.)*?"/, token: "string", next: "start"}, |
||||
{regex: /.*/, token: "string"} |
||||
], |
||||
string3: [ |
||||
{regex: /(?:[^\\]|\\.)*?"""/, token: "string", next: "start"}, |
||||
{regex: /.*/, token: "string"} |
||||
], |
||||
stack: [ |
||||
{regex: /\)/, token: "meta", next: "start"}, |
||||
{regex: /--/, token: "meta"}, |
||||
{regex: /\S+/, token: "variable-3"}, |
||||
{ |
||||
regex: /./, |
||||
token: null |
||||
} |
||||
], |
||||
// The meta property contains global information about the mode. It
|
||||
// can contain properties like lineComment, which are supported by
|
||||
// all modes, and also directives like dontIndentStates, which are
|
||||
// specific to simple modes.
|
||||
meta: { |
||||
dontIndentStates: ["start", "vocabulary", "string", "string3", "stack"], |
||||
lineComment: [ "!", "#!" ] |
||||
} |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-factor", "factor"); |
||||
}); |
@ -0,0 +1,77 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Factor mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link href='http://fonts.googleapis.com/css?family=Droid+Sans+Mono' rel='stylesheet' type='text/css'> |
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/mode/simple.js"></script> |
||||
<script src="factor.js"></script> |
||||
<style> |
||||
.CodeMirror { |
||||
font-family: 'Droid Sans Mono', monospace; |
||||
font-size: 14px; |
||||
} |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Factor</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
|
||||
<h2>Factor mode</h2> |
||||
|
||||
<form><textarea id="code" name="code"> |
||||
! Copyright (C) 2008 Slava Pestov. |
||||
! See http://factorcode.org/license.txt for BSD license. |
||||
|
||||
! A simple time server |
||||
|
||||
USING: accessors calendar calendar.format io io.encodings.ascii |
||||
io.servers kernel threads ; |
||||
IN: time-server |
||||
|
||||
: handle-time-client ( -- ) |
||||
now timestamp>rfc822 print ; |
||||
|
||||
: <time-server> ( -- threaded-server ) |
||||
ascii <threaded-server> |
||||
"time-server" >>name |
||||
1234 >>insecure |
||||
[ handle-time-client ] >>handler ; |
||||
|
||||
: start-time-server ( -- ) |
||||
<time-server> start-server drop ; |
||||
|
||||
MAIN: start-time-server |
||||
</textarea> |
||||
</form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
lineWrapping: true, |
||||
indentUnit: 2, |
||||
tabSize: 2, |
||||
autofocus: true, |
||||
mode: "text/x-factor" |
||||
}); |
||||
</script> |
||||
<p/> |
||||
<p>Simple mode that handles Factor Syntax (<a href="http://en.wikipedia.org/wiki/Factor_(programming_language)">Factor on WikiPedia</a>).</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-factor</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,173 @@ |
||||
// 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("fcl", function(config) { |
||||
var indentUnit = config.indentUnit; |
||||
|
||||
var keywords = { |
||||
"term": true, |
||||
"method": true, "accu": true, |
||||
"rule": true, "then": true, "is": true, "and": true, "or": true, |
||||
"if": true, "default": true |
||||
}; |
||||
|
||||
var start_blocks = { |
||||
"var_input": true, |
||||
"var_output": true, |
||||
"fuzzify": true, |
||||
"defuzzify": true, |
||||
"function_block": true, |
||||
"ruleblock": true |
||||
}; |
||||
|
||||
var end_blocks = { |
||||
"end_ruleblock": true, |
||||
"end_defuzzify": true, |
||||
"end_function_block": true, |
||||
"end_fuzzify": true, |
||||
"end_var": true |
||||
}; |
||||
|
||||
var atoms = { |
||||
"true": true, "false": true, "nan": true, |
||||
"real": true, "min": true, "max": true, "cog": true, "cogs": true |
||||
}; |
||||
|
||||
var isOperatorChar = /[+\-*&^%:=<>!|\/]/; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
|
||||
if (/[\d\.]/.test(ch)) { |
||||
if (ch == ".") { |
||||
stream.match(/^[0-9]+([eE][\-+]?[0-9]+)?/); |
||||
} else if (ch == "0") { |
||||
stream.match(/^[xX][0-9a-fA-F]+/) || stream.match(/^0[0-7]+/); |
||||
} else { |
||||
stream.match(/^[0-9]*\.?[0-9]*([eE][\-+]?[0-9]+)?/); |
||||
} |
||||
return "number"; |
||||
} |
||||
|
||||
if (ch == "/" || ch == "(") { |
||||
if (stream.eat("*")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenComment(stream, state); |
||||
} |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
|
||||
var cur = stream.current().toLowerCase(); |
||||
if (keywords.propertyIsEnumerable(cur) || |
||||
start_blocks.propertyIsEnumerable(cur) || |
||||
end_blocks.propertyIsEnumerable(cur)) { |
||||
return "keyword"; |
||||
} |
||||
if (atoms.propertyIsEnumerable(cur)) return "atom"; |
||||
return "variable"; |
||||
} |
||||
|
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if ((ch == "/" || ch == ")") && maybeEnd) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
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) { |
||||
if (!state.context.prev) return; |
||||
var t = state.context.type; |
||||
if (t == "end_block") |
||||
state.indented = state.context.indented; |
||||
return state.context = state.context.prev; |
||||
} |
||||
|
||||
// Interface
|
||||
|
||||
return { |
||||
startState: function(basecolumn) { |
||||
return { |
||||
tokenize: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), |
||||
indented: 0, |
||||
startOfLine: true |
||||
}; |
||||
}, |
||||
|
||||
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; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
|
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
var cur = stream.current().toLowerCase(); |
||||
|
||||
if (start_blocks.propertyIsEnumerable(cur)) pushContext(state, stream.column(), "end_block"); |
||||
else if (end_blocks.propertyIsEnumerable(cur)) popContext(state); |
||||
|
||||
state.startOfLine = false; |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != tokenBase && state.tokenize != null) return 0; |
||||
var ctx = state.context; |
||||
|
||||
var closing = end_blocks.propertyIsEnumerable(textAfter); |
||||
if (ctx.align) return ctx.column + (closing ? 0 : 1); |
||||
else return ctx.indented + (closing ? 0 : indentUnit); |
||||
}, |
||||
|
||||
electricChars: "ryk", |
||||
fold: "brace", |
||||
blockCommentStart: "(*", |
||||
blockCommentEnd: "*)", |
||||
lineComment: "//" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-fcl", "fcl"); |
||||
}); |
@ -0,0 +1,108 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: FCL mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/elegant.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="fcl.js"></script> |
||||
<style>.CodeMirror {border:1px solid #999; background:#ffc}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">FCL</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>FCL mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
FUNCTION_BLOCK Fuzzy_FB |
||||
VAR_INPUT |
||||
TimeDay : REAL; (* RANGE(0 .. 23) *) |
||||
ApplicateHost: REAL; |
||||
TimeConfiguration: REAL; |
||||
TimeRequirements: REAL; |
||||
END_VAR |
||||
|
||||
VAR_OUTPUT |
||||
ProbabilityDistribution: REAL; |
||||
ProbabilityAccess: REAL; |
||||
END_VAR |
||||
|
||||
FUZZIFY TimeDay |
||||
TERM inside := (0, 0) (8, 1) (22,0); |
||||
TERM outside := (0, 1) (8, 0) (22, 1); |
||||
END_FUZZIFY |
||||
|
||||
FUZZIFY ApplicateHost |
||||
TERM few := (0, 1) (100, 0) (200, 0); |
||||
TERM many := (0, 0) (100, 0) (200, 1); |
||||
END_FUZZIFY |
||||
|
||||
FUZZIFY TimeConfiguration |
||||
TERM recently := (0, 1) (30, 1) (120, 0); |
||||
TERM long := (0, 0) (30, 0) (120, 1); |
||||
END_FUZZIFY |
||||
|
||||
FUZZIFY TimeRequirements |
||||
TERM recently := (0, 1) (30, 1) (365, 0); |
||||
TERM long := (0, 0) (30, 0) (365, 1); |
||||
END_FUZZIFY |
||||
|
||||
DEFUZZIFY ProbabilityAccess |
||||
TERM hight := 1; |
||||
TERM medium := 0.5; |
||||
TERM low := 0; |
||||
ACCU: MAX; |
||||
METHOD: COGS; |
||||
DEFAULT := 0; |
||||
END_DEFUZZIFY |
||||
|
||||
DEFUZZIFY ProbabilityDistribution |
||||
TERM hight := 1; |
||||
TERM medium := 0.5; |
||||
TERM low := 0; |
||||
ACCU: MAX; |
||||
METHOD: COGS; |
||||
DEFAULT := 0; |
||||
END_DEFUZZIFY |
||||
|
||||
RULEBLOCK No1 |
||||
AND : MIN; |
||||
RULE 1 : IF TimeDay IS outside AND ApplicateHost IS few THEN ProbabilityAccess IS hight; |
||||
RULE 2 : IF ApplicateHost IS many THEN ProbabilityAccess IS hight; |
||||
RULE 3 : IF TimeDay IS inside AND ApplicateHost IS few THEN ProbabilityAccess IS low; |
||||
END_RULEBLOCK |
||||
|
||||
RULEBLOCK No2 |
||||
AND : MIN; |
||||
RULE 1 : IF ApplicateHost IS many THEN ProbabilityDistribution IS hight; |
||||
END_RULEBLOCK |
||||
|
||||
END_FUNCTION_BLOCK |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
theme: "elegant", |
||||
matchBrackets: true, |
||||
indentUnit: 8, |
||||
tabSize: 8, |
||||
indentWithTabs: true, |
||||
mode: "text/x-fcl" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME type:</strong> <code>text/x-fcl</code></p> |
||||
</article> |
@ -0,0 +1,180 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
// Author: Aliaksei Chapyzhenka
|
||||
|
||||
(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"; |
||||
|
||||
function toWordList(words) { |
||||
var ret = []; |
||||
words.split(' ').forEach(function(e){ |
||||
ret.push({name: e}); |
||||
}); |
||||
return ret; |
||||
} |
||||
|
||||
var coreWordList = toWordList( |
||||
'INVERT AND OR XOR\ |
||||
2* 2/ LSHIFT RSHIFT\ |
||||
0= = 0< < > U< MIN MAX\ |
||||
2DROP 2DUP 2OVER 2SWAP ?DUP DEPTH DROP DUP OVER ROT SWAP\ |
||||
>R R> R@\ |
||||
+ - 1+ 1- ABS NEGATE\ |
||||
S>D * M* UM*\ |
||||
FM/MOD SM/REM UM/MOD */ */MOD / /MOD MOD\ |
||||
HERE , @ ! CELL+ CELLS C, C@ C! CHARS 2@ 2!\ |
||||
ALIGN ALIGNED +! ALLOT\ |
||||
CHAR [CHAR] [ ] BL\ |
||||
FIND EXECUTE IMMEDIATE COUNT LITERAL STATE\ |
||||
; DOES> >BODY\ |
||||
EVALUATE\ |
||||
SOURCE >IN\ |
||||
<# # #S #> HOLD SIGN BASE >NUMBER HEX DECIMAL\ |
||||
FILL MOVE\ |
||||
. CR EMIT SPACE SPACES TYPE U. .R U.R\ |
||||
ACCEPT\ |
||||
TRUE FALSE\ |
||||
<> U> 0<> 0>\ |
||||
NIP TUCK ROLL PICK\ |
||||
2>R 2R@ 2R>\ |
||||
WITHIN UNUSED MARKER\ |
||||
I J\ |
||||
TO\ |
||||
COMPILE, [COMPILE]\ |
||||
SAVE-INPUT RESTORE-INPUT\ |
||||
PAD ERASE\ |
||||
2LITERAL DNEGATE\ |
||||
D- D+ D0< D0= D2* D2/ D< D= DMAX DMIN D>S DABS\ |
||||
M+ M*/ D. D.R 2ROT DU<\ |
||||
CATCH THROW\ |
||||
FREE RESIZE ALLOCATE\ |
||||
CS-PICK CS-ROLL\ |
||||
GET-CURRENT SET-CURRENT FORTH-WORDLIST GET-ORDER SET-ORDER\ |
||||
PREVIOUS SEARCH-WORDLIST WORDLIST FIND ALSO ONLY FORTH DEFINITIONS ORDER\ |
||||
-TRAILING /STRING SEARCH COMPARE CMOVE CMOVE> BLANK SLITERAL'); |
||||
|
||||
var immediateWordList = toWordList('IF ELSE THEN BEGIN WHILE REPEAT UNTIL RECURSE [IF] [ELSE] [THEN] ?DO DO LOOP +LOOP UNLOOP LEAVE EXIT AGAIN CASE OF ENDOF ENDCASE'); |
||||
|
||||
CodeMirror.defineMode('forth', function() { |
||||
function searchWordList (wordList, word) { |
||||
var i; |
||||
for (i = wordList.length - 1; i >= 0; i--) { |
||||
if (wordList[i].name === word.toUpperCase()) { |
||||
return wordList[i]; |
||||
} |
||||
} |
||||
return undefined; |
||||
} |
||||
return { |
||||
startState: function() { |
||||
return { |
||||
state: '', |
||||
base: 10, |
||||
coreWordList: coreWordList, |
||||
immediateWordList: immediateWordList, |
||||
wordList: [] |
||||
}; |
||||
}, |
||||
token: function (stream, stt) { |
||||
var mat; |
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
if (stt.state === '') { // interpretation
|
||||
if (stream.match(/^(\]|:NONAME)(\s|$)/i)) { |
||||
stt.state = ' compilation'; |
||||
return 'builtin compilation'; |
||||
} |
||||
mat = stream.match(/^(\:)\s+(\S+)(\s|$)+/); |
||||
if (mat) { |
||||
stt.wordList.push({name: mat[2].toUpperCase()}); |
||||
stt.state = ' compilation'; |
||||
return 'def' + stt.state; |
||||
} |
||||
mat = stream.match(/^(VARIABLE|2VARIABLE|CONSTANT|2CONSTANT|CREATE|POSTPONE|VALUE|WORD)\s+(\S+)(\s|$)+/i); |
||||
if (mat) { |
||||
stt.wordList.push({name: mat[2].toUpperCase()}); |
||||
return 'def' + stt.state; |
||||
} |
||||
mat = stream.match(/^(\'|\[\'\])\s+(\S+)(\s|$)+/); |
||||
if (mat) { |
||||
return 'builtin' + stt.state; |
||||
} |
||||
} else { // compilation
|
||||
// ; [
|
||||
if (stream.match(/^(\;|\[)(\s)/)) { |
||||
stt.state = ''; |
||||
stream.backUp(1); |
||||
return 'builtin compilation'; |
||||
} |
||||
if (stream.match(/^(\;|\[)($)/)) { |
||||
stt.state = ''; |
||||
return 'builtin compilation'; |
||||
} |
||||
if (stream.match(/^(POSTPONE)\s+\S+(\s|$)+/)) { |
||||
return 'builtin'; |
||||
} |
||||
} |
||||
|
||||
// dynamic wordlist
|
||||
mat = stream.match(/^(\S+)(\s+|$)/); |
||||
if (mat) { |
||||
if (searchWordList(stt.wordList, mat[1]) !== undefined) { |
||||
return 'variable' + stt.state; |
||||
} |
||||
|
||||
// comments
|
||||
if (mat[1] === '\\') { |
||||
stream.skipToEnd(); |
||||
return 'comment' + stt.state; |
||||
} |
||||
|
||||
// core words
|
||||
if (searchWordList(stt.coreWordList, mat[1]) !== undefined) { |
||||
return 'builtin' + stt.state; |
||||
} |
||||
if (searchWordList(stt.immediateWordList, mat[1]) !== undefined) { |
||||
return 'keyword' + stt.state; |
||||
} |
||||
|
||||
if (mat[1] === '(') { |
||||
stream.eatWhile(function (s) { return s !== ')'; }); |
||||
stream.eat(')'); |
||||
return 'comment' + stt.state; |
||||
} |
||||
|
||||
// // strings
|
||||
if (mat[1] === '.(') { |
||||
stream.eatWhile(function (s) { return s !== ')'; }); |
||||
stream.eat(')'); |
||||
return 'string' + stt.state; |
||||
} |
||||
if (mat[1] === 'S"' || mat[1] === '."' || mat[1] === 'C"') { |
||||
stream.eatWhile(function (s) { return s !== '"'; }); |
||||
stream.eat('"'); |
||||
return 'string' + stt.state; |
||||
} |
||||
|
||||
// numbers
|
||||
if (mat[1] - 0xfffffffff) { |
||||
return 'number' + stt.state; |
||||
} |
||||
// if (mat[1].match(/^[-+]?[0-9]+\.[0-9]*/)) {
|
||||
// return 'number' + stt.state;
|
||||
// }
|
||||
|
||||
return 'atom' + stt.state; |
||||
} |
||||
} |
||||
}; |
||||
}); |
||||
CodeMirror.defineMIME("text/x-forth", "forth"); |
||||
}); |
@ -0,0 +1,75 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Forth mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link href='http://fonts.googleapis.com/css?family=Droid+Sans+Mono' rel='stylesheet' type='text/css'> |
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel=stylesheet href="../../theme/colorforth.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="forth.js"></script> |
||||
<style> |
||||
.CodeMirror { |
||||
font-family: 'Droid Sans Mono', monospace; |
||||
font-size: 14px; |
||||
} |
||||
</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Forth</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
|
||||
<h2>Forth mode</h2> |
||||
|
||||
<form><textarea id="code" name="code"> |
||||
\ Insertion sort |
||||
|
||||
: cell- 1 cells - ; |
||||
|
||||
: insert ( start end -- start ) |
||||
dup @ >r ( r: v ) |
||||
begin |
||||
2dup < |
||||
while |
||||
r@ over cell- @ < |
||||
while |
||||
cell- |
||||
dup @ over cell+ ! |
||||
repeat then |
||||
r> swap ! ; |
||||
|
||||
: sort ( array len -- ) |
||||
1 ?do |
||||
dup i cells + insert |
||||
loop drop ;</textarea> |
||||
</form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
lineWrapping: true, |
||||
indentUnit: 2, |
||||
tabSize: 2, |
||||
autofocus: true, |
||||
theme: "colorforth", |
||||
mode: "text/x-forth" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Simple mode that handle Forth-Syntax (<a href="http://en.wikipedia.org/wiki/Forth_%28programming_language%29">Forth on WikiPedia</a>).</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-forth</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,188 @@ |
||||
// 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("fortran", function() { |
||||
function words(array) { |
||||
var keys = {}; |
||||
for (var i = 0; i < array.length; ++i) { |
||||
keys[array[i]] = true; |
||||
} |
||||
return keys; |
||||
} |
||||
|
||||
var keywords = words([ |
||||
"abstract", "accept", "allocatable", "allocate", |
||||
"array", "assign", "asynchronous", "backspace", |
||||
"bind", "block", "byte", "call", "case", |
||||
"class", "close", "common", "contains", |
||||
"continue", "cycle", "data", "deallocate", |
||||
"decode", "deferred", "dimension", "do", |
||||
"elemental", "else", "encode", "end", |
||||
"endif", "entry", "enumerator", "equivalence", |
||||
"exit", "external", "extrinsic", "final", |
||||
"forall", "format", "function", "generic", |
||||
"go", "goto", "if", "implicit", "import", "include", |
||||
"inquire", "intent", "interface", "intrinsic", |
||||
"module", "namelist", "non_intrinsic", |
||||
"non_overridable", "none", "nopass", |
||||
"nullify", "open", "optional", "options", |
||||
"parameter", "pass", "pause", "pointer", |
||||
"print", "private", "program", "protected", |
||||
"public", "pure", "read", "recursive", "result", |
||||
"return", "rewind", "save", "select", "sequence", |
||||
"stop", "subroutine", "target", "then", "to", "type", |
||||
"use", "value", "volatile", "where", "while", |
||||
"write"]); |
||||
var builtins = words(["abort", "abs", "access", "achar", "acos", |
||||
"adjustl", "adjustr", "aimag", "aint", "alarm", |
||||
"all", "allocated", "alog", "amax", "amin", |
||||
"amod", "and", "anint", "any", "asin", |
||||
"associated", "atan", "besj", "besjn", "besy", |
||||
"besyn", "bit_size", "btest", "cabs", "ccos", |
||||
"ceiling", "cexp", "char", "chdir", "chmod", |
||||
"clog", "cmplx", "command_argument_count", |
||||
"complex", "conjg", "cos", "cosh", "count", |
||||
"cpu_time", "cshift", "csin", "csqrt", "ctime", |
||||
"c_funloc", "c_loc", "c_associated", "c_null_ptr", |
||||
"c_null_funptr", "c_f_pointer", "c_null_char", |
||||
"c_alert", "c_backspace", "c_form_feed", |
||||
"c_new_line", "c_carriage_return", |
||||
"c_horizontal_tab", "c_vertical_tab", "dabs", |
||||
"dacos", "dasin", "datan", "date_and_time", |
||||
"dbesj", "dbesj", "dbesjn", "dbesy", "dbesy", |
||||
"dbesyn", "dble", "dcos", "dcosh", "ddim", "derf", |
||||
"derfc", "dexp", "digits", "dim", "dint", "dlog", |
||||
"dlog", "dmax", "dmin", "dmod", "dnint", |
||||
"dot_product", "dprod", "dsign", "dsinh", |
||||
"dsin", "dsqrt", "dtanh", "dtan", "dtime", |
||||
"eoshift", "epsilon", "erf", "erfc", "etime", |
||||
"exit", "exp", "exponent", "extends_type_of", |
||||
"fdate", "fget", "fgetc", "float", "floor", |
||||
"flush", "fnum", "fputc", "fput", "fraction", |
||||
"fseek", "fstat", "ftell", "gerror", "getarg", |
||||
"get_command", "get_command_argument", |
||||
"get_environment_variable", "getcwd", |
||||
"getenv", "getgid", "getlog", "getpid", |
||||
"getuid", "gmtime", "hostnm", "huge", "iabs", |
||||
"iachar", "iand", "iargc", "ibclr", "ibits", |
||||
"ibset", "ichar", "idate", "idim", "idint", |
||||
"idnint", "ieor", "ierrno", "ifix", "imag", |
||||
"imagpart", "index", "int", "ior", "irand", |
||||
"isatty", "ishft", "ishftc", "isign", |
||||
"iso_c_binding", "is_iostat_end", "is_iostat_eor", |
||||
"itime", "kill", "kind", "lbound", "len", "len_trim", |
||||
"lge", "lgt", "link", "lle", "llt", "lnblnk", "loc", |
||||
"log", "logical", "long", "lshift", "lstat", "ltime", |
||||
"matmul", "max", "maxexponent", "maxloc", "maxval", |
||||
"mclock", "merge", "move_alloc", "min", "minexponent", |
||||
"minloc", "minval", "mod", "modulo", "mvbits", |
||||
"nearest", "new_line", "nint", "not", "or", "pack", |
||||
"perror", "precision", "present", "product", "radix", |
||||
"rand", "random_number", "random_seed", "range", |
||||
"real", "realpart", "rename", "repeat", "reshape", |
||||
"rrspacing", "rshift", "same_type_as", "scale", |
||||
"scan", "second", "selected_int_kind", |
||||
"selected_real_kind", "set_exponent", "shape", |
||||
"short", "sign", "signal", "sinh", "sin", "sleep", |
||||
"sngl", "spacing", "spread", "sqrt", "srand", "stat", |
||||
"sum", "symlnk", "system", "system_clock", "tan", |
||||
"tanh", "time", "tiny", "transfer", "transpose", |
||||
"trim", "ttynam", "ubound", "umask", "unlink", |
||||
"unpack", "verify", "xor", "zabs", "zcos", "zexp", |
||||
"zlog", "zsin", "zsqrt"]); |
||||
|
||||
var dataTypes = words(["c_bool", "c_char", "c_double", "c_double_complex", |
||||
"c_float", "c_float_complex", "c_funptr", "c_int", |
||||
"c_int16_t", "c_int32_t", "c_int64_t", "c_int8_t", |
||||
"c_int_fast16_t", "c_int_fast32_t", "c_int_fast64_t", |
||||
"c_int_fast8_t", "c_int_least16_t", "c_int_least32_t", |
||||
"c_int_least64_t", "c_int_least8_t", "c_intmax_t", |
||||
"c_intptr_t", "c_long", "c_long_double", |
||||
"c_long_double_complex", "c_long_long", "c_ptr", |
||||
"c_short", "c_signed_char", "c_size_t", "character", |
||||
"complex", "double", "integer", "logical", "real"]); |
||||
var isOperatorChar = /[+\-*&=<>\/\:]/; |
||||
var litOperator = new RegExp("(\.and\.|\.or\.|\.eq\.|\.lt\.|\.le\.|\.gt\.|\.ge\.|\.ne\.|\.not\.|\.eqv\.|\.neqv\.)", "i"); |
||||
|
||||
function tokenBase(stream, state) { |
||||
|
||||
if (stream.match(litOperator)){ |
||||
return 'operator'; |
||||
} |
||||
|
||||
var ch = stream.next(); |
||||
if (ch == "!") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
if (ch == '"' || ch == "'") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (/[\[\]\(\),]/.test(ch)) { |
||||
return null; |
||||
} |
||||
if (/\d/.test(ch)) { |
||||
stream.eatWhile(/[\w\.]/); |
||||
return "number"; |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_]/); |
||||
var word = stream.current().toLowerCase(); |
||||
|
||||
if (keywords.hasOwnProperty(word)){ |
||||
return 'keyword'; |
||||
} |
||||
if (builtins.hasOwnProperty(word) || dataTypes.hasOwnProperty(word)) { |
||||
return 'builtin'; |
||||
} |
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) { |
||||
end = true; |
||||
break; |
||||
} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
if (end || !escaped) state.tokenize = null; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
// Interface
|
||||
|
||||
return { |
||||
startState: function() { |
||||
return {tokenize: null}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (stream.eatSpace()) return null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment" || style == "meta") return style; |
||||
return style; |
||||
} |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-fortran", "fortran"); |
||||
|
||||
}); |
@ -0,0 +1,81 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Fortran mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="fortran.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Fortran</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Fortran mode</h2> |
||||
|
||||
|
||||
<div><textarea id="code" name="code"> |
||||
! Example Fortran code |
||||
program average |
||||
|
||||
! Read in some numbers and take the average |
||||
! As written, if there are no data points, an average of zero is returned |
||||
! While this may not be desired behavior, it keeps this example simple |
||||
|
||||
implicit none |
||||
|
||||
real, dimension(:), allocatable :: points |
||||
integer :: number_of_points |
||||
real :: average_points=0., positive_average=0., negative_average=0. |
||||
|
||||
write (*,*) "Input number of points to average:" |
||||
read (*,*) number_of_points |
||||
|
||||
allocate (points(number_of_points)) |
||||
|
||||
write (*,*) "Enter the points to average:" |
||||
read (*,*) points |
||||
|
||||
! Take the average by summing points and dividing by number_of_points |
||||
if (number_of_points > 0) average_points = sum(points) / number_of_points |
||||
|
||||
! Now form average over positive and negative points only |
||||
if (count(points > 0.) > 0) then |
||||
positive_average = sum(points, points > 0.) / count(points > 0.) |
||||
end if |
||||
|
||||
if (count(points < 0.) > 0) then |
||||
negative_average = sum(points, points < 0.) / count(points < 0.) |
||||
end if |
||||
|
||||
deallocate (points) |
||||
|
||||
! Print result to terminal |
||||
write (*,'(a,g12.4)') 'Average = ', average_points |
||||
write (*,'(a,g12.4)') 'Average of positive points = ', positive_average |
||||
write (*,'(a,g12.4)') 'Average of negative points = ', negative_average |
||||
|
||||
end program average |
||||
</textarea></div> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "text/x-fortran" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-fortran</code>.</p> |
||||
</article> |
@ -0,0 +1,345 @@ |
||||
// 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("gas", function(_config, parserConfig) { |
||||
'use strict'; |
||||
|
||||
// If an architecture is specified, its initialization function may
|
||||
// populate this array with custom parsing functions which will be
|
||||
// tried in the event that the standard functions do not find a match.
|
||||
var custom = []; |
||||
|
||||
// The symbol used to start a line comment changes based on the target
|
||||
// architecture.
|
||||
// If no architecture is pased in "parserConfig" then only multiline
|
||||
// comments will have syntax support.
|
||||
var lineCommentStartSymbol = ""; |
||||
|
||||
// These directives are architecture independent.
|
||||
// Machine specific directives should go in their respective
|
||||
// architecture initialization function.
|
||||
// Reference:
|
||||
// http://sourceware.org/binutils/docs/as/Pseudo-Ops.html#Pseudo-Ops
|
||||
var directives = { |
||||
".abort" : "builtin", |
||||
".align" : "builtin", |
||||
".altmacro" : "builtin", |
||||
".ascii" : "builtin", |
||||
".asciz" : "builtin", |
||||
".balign" : "builtin", |
||||
".balignw" : "builtin", |
||||
".balignl" : "builtin", |
||||
".bundle_align_mode" : "builtin", |
||||
".bundle_lock" : "builtin", |
||||
".bundle_unlock" : "builtin", |
||||
".byte" : "builtin", |
||||
".cfi_startproc" : "builtin", |
||||
".comm" : "builtin", |
||||
".data" : "builtin", |
||||
".def" : "builtin", |
||||
".desc" : "builtin", |
||||
".dim" : "builtin", |
||||
".double" : "builtin", |
||||
".eject" : "builtin", |
||||
".else" : "builtin", |
||||
".elseif" : "builtin", |
||||
".end" : "builtin", |
||||
".endef" : "builtin", |
||||
".endfunc" : "builtin", |
||||
".endif" : "builtin", |
||||
".equ" : "builtin", |
||||
".equiv" : "builtin", |
||||
".eqv" : "builtin", |
||||
".err" : "builtin", |
||||
".error" : "builtin", |
||||
".exitm" : "builtin", |
||||
".extern" : "builtin", |
||||
".fail" : "builtin", |
||||
".file" : "builtin", |
||||
".fill" : "builtin", |
||||
".float" : "builtin", |
||||
".func" : "builtin", |
||||
".global" : "builtin", |
||||
".gnu_attribute" : "builtin", |
||||
".hidden" : "builtin", |
||||
".hword" : "builtin", |
||||
".ident" : "builtin", |
||||
".if" : "builtin", |
||||
".incbin" : "builtin", |
||||
".include" : "builtin", |
||||
".int" : "builtin", |
||||
".internal" : "builtin", |
||||
".irp" : "builtin", |
||||
".irpc" : "builtin", |
||||
".lcomm" : "builtin", |
||||
".lflags" : "builtin", |
||||
".line" : "builtin", |
||||
".linkonce" : "builtin", |
||||
".list" : "builtin", |
||||
".ln" : "builtin", |
||||
".loc" : "builtin", |
||||
".loc_mark_labels" : "builtin", |
||||
".local" : "builtin", |
||||
".long" : "builtin", |
||||
".macro" : "builtin", |
||||
".mri" : "builtin", |
||||
".noaltmacro" : "builtin", |
||||
".nolist" : "builtin", |
||||
".octa" : "builtin", |
||||
".offset" : "builtin", |
||||
".org" : "builtin", |
||||
".p2align" : "builtin", |
||||
".popsection" : "builtin", |
||||
".previous" : "builtin", |
||||
".print" : "builtin", |
||||
".protected" : "builtin", |
||||
".psize" : "builtin", |
||||
".purgem" : "builtin", |
||||
".pushsection" : "builtin", |
||||
".quad" : "builtin", |
||||
".reloc" : "builtin", |
||||
".rept" : "builtin", |
||||
".sbttl" : "builtin", |
||||
".scl" : "builtin", |
||||
".section" : "builtin", |
||||
".set" : "builtin", |
||||
".short" : "builtin", |
||||
".single" : "builtin", |
||||
".size" : "builtin", |
||||
".skip" : "builtin", |
||||
".sleb128" : "builtin", |
||||
".space" : "builtin", |
||||
".stab" : "builtin", |
||||
".string" : "builtin", |
||||
".struct" : "builtin", |
||||
".subsection" : "builtin", |
||||
".symver" : "builtin", |
||||
".tag" : "builtin", |
||||
".text" : "builtin", |
||||
".title" : "builtin", |
||||
".type" : "builtin", |
||||
".uleb128" : "builtin", |
||||
".val" : "builtin", |
||||
".version" : "builtin", |
||||
".vtable_entry" : "builtin", |
||||
".vtable_inherit" : "builtin", |
||||
".warning" : "builtin", |
||||
".weak" : "builtin", |
||||
".weakref" : "builtin", |
||||
".word" : "builtin" |
||||
}; |
||||
|
||||
var registers = {}; |
||||
|
||||
function x86(_parserConfig) { |
||||
lineCommentStartSymbol = "#"; |
||||
|
||||
registers.ax = "variable"; |
||||
registers.eax = "variable-2"; |
||||
registers.rax = "variable-3"; |
||||
|
||||
registers.bx = "variable"; |
||||
registers.ebx = "variable-2"; |
||||
registers.rbx = "variable-3"; |
||||
|
||||
registers.cx = "variable"; |
||||
registers.ecx = "variable-2"; |
||||
registers.rcx = "variable-3"; |
||||
|
||||
registers.dx = "variable"; |
||||
registers.edx = "variable-2"; |
||||
registers.rdx = "variable-3"; |
||||
|
||||
registers.si = "variable"; |
||||
registers.esi = "variable-2"; |
||||
registers.rsi = "variable-3"; |
||||
|
||||
registers.di = "variable"; |
||||
registers.edi = "variable-2"; |
||||
registers.rdi = "variable-3"; |
||||
|
||||
registers.sp = "variable"; |
||||
registers.esp = "variable-2"; |
||||
registers.rsp = "variable-3"; |
||||
|
||||
registers.bp = "variable"; |
||||
registers.ebp = "variable-2"; |
||||
registers.rbp = "variable-3"; |
||||
|
||||
registers.ip = "variable"; |
||||
registers.eip = "variable-2"; |
||||
registers.rip = "variable-3"; |
||||
|
||||
registers.cs = "keyword"; |
||||
registers.ds = "keyword"; |
||||
registers.ss = "keyword"; |
||||
registers.es = "keyword"; |
||||
registers.fs = "keyword"; |
||||
registers.gs = "keyword"; |
||||
} |
||||
|
||||
function armv6(_parserConfig) { |
||||
// Reference:
|
||||
// http://infocenter.arm.com/help/topic/com.arm.doc.qrc0001l/QRC0001_UAL.pdf
|
||||
// http://infocenter.arm.com/help/topic/com.arm.doc.ddi0301h/DDI0301H_arm1176jzfs_r0p7_trm.pdf
|
||||
lineCommentStartSymbol = "@"; |
||||
directives.syntax = "builtin"; |
||||
|
||||
registers.r0 = "variable"; |
||||
registers.r1 = "variable"; |
||||
registers.r2 = "variable"; |
||||
registers.r3 = "variable"; |
||||
registers.r4 = "variable"; |
||||
registers.r5 = "variable"; |
||||
registers.r6 = "variable"; |
||||
registers.r7 = "variable"; |
||||
registers.r8 = "variable"; |
||||
registers.r9 = "variable"; |
||||
registers.r10 = "variable"; |
||||
registers.r11 = "variable"; |
||||
registers.r12 = "variable"; |
||||
|
||||
registers.sp = "variable-2"; |
||||
registers.lr = "variable-2"; |
||||
registers.pc = "variable-2"; |
||||
registers.r13 = registers.sp; |
||||
registers.r14 = registers.lr; |
||||
registers.r15 = registers.pc; |
||||
|
||||
custom.push(function(ch, stream) { |
||||
if (ch === '#') { |
||||
stream.eatWhile(/\w/); |
||||
return "number"; |
||||
} |
||||
}); |
||||
} |
||||
|
||||
var arch = (parserConfig.architecture || "x86").toLowerCase(); |
||||
if (arch === "x86") { |
||||
x86(parserConfig); |
||||
} else if (arch === "arm" || arch === "armv6") { |
||||
armv6(parserConfig); |
||||
} |
||||
|
||||
function nextUntilUnescaped(stream, end) { |
||||
var escaped = false, next; |
||||
while ((next = stream.next()) != null) { |
||||
if (next === end && !escaped) { |
||||
return false; |
||||
} |
||||
escaped = !escaped && next === "\\"; |
||||
} |
||||
return escaped; |
||||
} |
||||
|
||||
function clikeComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while ((ch = stream.next()) != null) { |
||||
if (ch === "/" && maybeEnd) { |
||||
state.tokenize = null; |
||||
break; |
||||
} |
||||
maybeEnd = (ch === "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
return { |
||||
startState: function() { |
||||
return { |
||||
tokenize: null |
||||
}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (state.tokenize) { |
||||
return state.tokenize(stream, state); |
||||
} |
||||
|
||||
if (stream.eatSpace()) { |
||||
return null; |
||||
} |
||||
|
||||
var style, cur, ch = stream.next(); |
||||
|
||||
if (ch === "/") { |
||||
if (stream.eat("*")) { |
||||
state.tokenize = clikeComment; |
||||
return clikeComment(stream, state); |
||||
} |
||||
} |
||||
|
||||
if (ch === lineCommentStartSymbol) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
|
||||
if (ch === '"') { |
||||
nextUntilUnescaped(stream, '"'); |
||||
return "string"; |
||||
} |
||||
|
||||
if (ch === '.') { |
||||
stream.eatWhile(/\w/); |
||||
cur = stream.current().toLowerCase(); |
||||
style = directives[cur]; |
||||
return style || null; |
||||
} |
||||
|
||||
if (ch === '=') { |
||||
stream.eatWhile(/\w/); |
||||
return "tag"; |
||||
} |
||||
|
||||
if (ch === '{') { |
||||
return "braket"; |
||||
} |
||||
|
||||
if (ch === '}') { |
||||
return "braket"; |
||||
} |
||||
|
||||
if (/\d/.test(ch)) { |
||||
if (ch === "0" && stream.eat("x")) { |
||||
stream.eatWhile(/[0-9a-fA-F]/); |
||||
return "number"; |
||||
} |
||||
stream.eatWhile(/\d/); |
||||
return "number"; |
||||
} |
||||
|
||||
if (/\w/.test(ch)) { |
||||
stream.eatWhile(/\w/); |
||||
if (stream.eat(":")) { |
||||
return 'tag'; |
||||
} |
||||
cur = stream.current().toLowerCase(); |
||||
style = registers[cur]; |
||||
return style || null; |
||||
} |
||||
|
||||
for (var i = 0; i < custom.length; i++) { |
||||
style = custom[i](ch, stream, state); |
||||
if (style) { |
||||
return style; |
||||
} |
||||
} |
||||
}, |
||||
|
||||
lineComment: lineCommentStartSymbol, |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/" |
||||
}; |
||||
}); |
||||
|
||||
}); |
@ -0,0 +1,68 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Gas mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="gas.js"></script> |
||||
<style>.CodeMirror {border: 2px inset #dee;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Gas</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Gas mode</h2> |
||||
<form> |
||||
<textarea id="code" name="code"> |
||||
.syntax unified |
||||
.global main |
||||
|
||||
/* |
||||
* A |
||||
* multi-line |
||||
* comment. |
||||
*/ |
||||
|
||||
@ A single line comment. |
||||
|
||||
main: |
||||
push {sp, lr} |
||||
ldr r0, =message |
||||
bl puts |
||||
mov r0, #0 |
||||
pop {sp, pc} |
||||
|
||||
message: |
||||
.asciz "Hello world!<br />" |
||||
</textarea> |
||||
</form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: {name: "gas", architecture: "ARMv6"}, |
||||
}); |
||||
</script> |
||||
|
||||
<p>Handles AT&T assembler syntax (more specifically this handles |
||||
the GNU Assembler (gas) syntax.) |
||||
It takes a single optional configuration parameter: |
||||
<code>architecture</code>, which can be one of <code>"ARM"</code>, |
||||
<code>"ARMv6"</code> or <code>"x86"</code>. |
||||
Including the parameter adds syntax for the registers and special |
||||
directives for the supplied architecture. |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-gas</code></p> |
||||
</article> |
@ -0,0 +1,130 @@ |
||||
// 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"), require("../markdown/markdown"), require("../../addon/mode/overlay")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../markdown/markdown", "../../addon/mode/overlay"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
var urlRE = /^((?:(?:aaas?|about|acap|adiumxtra|af[ps]|aim|apt|attachment|aw|beshare|bitcoin|bolo|callto|cap|chrome(?:-extension)?|cid|coap|com-eventbrite-attendee|content|crid|cvs|data|dav|dict|dlna-(?:playcontainer|playsingle)|dns|doi|dtn|dvb|ed2k|facetime|feed|file|finger|fish|ftp|geo|gg|git|gizmoproject|go|gopher|gtalk|h323|hcp|https?|iax|icap|icon|im|imap|info|ipn|ipp|irc[6s]?|iris(?:\.beep|\.lwz|\.xpc|\.xpcs)?|itms|jar|javascript|jms|keyparc|lastfm|ldaps?|magnet|mailto|maps|market|message|mid|mms|ms-help|msnim|msrps?|mtqp|mumble|mupdate|mvn|news|nfs|nih?|nntp|notes|oid|opaquelocktoken|palm|paparazzi|platform|pop|pres|proxy|psyc|query|res(?:ource)?|rmi|rsync|rtmp|rtsp|secondlife|service|session|sftp|sgn|shttp|sieve|sips?|skype|sm[bs]|snmp|soap\.beeps?|soldat|spotify|ssh|steam|svn|tag|teamspeak|tel(?:net)?|tftp|things|thismessage|tip|tn3270|tv|udp|unreal|urn|ut2004|vemmi|ventrilo|view-source|webcal|wss?|wtai|wyciwyg|xcon(?:-userid)?|xfire|xmlrpc\.beeps?|xmpp|xri|ymsgr|z39\.50[rs]?):(?:\/{1,3}|[a-z0-9%])|www\d{0,3}[.]|[a-z0-9.\-]+[.][a-z]{2,4}\/)(?:[^\s()<>]|\([^\s()<>]*\))+(?:\([^\s()<>]*\)|[^\s`*!()\[\]{};:'".,<>?«»“”‘’]))/i |
||||
|
||||
CodeMirror.defineMode("gfm", function(config, modeConfig) { |
||||
var codeDepth = 0; |
||||
function blankLine(state) { |
||||
state.code = false; |
||||
return null; |
||||
} |
||||
var gfmOverlay = { |
||||
startState: function() { |
||||
return { |
||||
code: false, |
||||
codeBlock: false, |
||||
ateSpace: false |
||||
}; |
||||
}, |
||||
copyState: function(s) { |
||||
return { |
||||
code: s.code, |
||||
codeBlock: s.codeBlock, |
||||
ateSpace: s.ateSpace |
||||
}; |
||||
}, |
||||
token: function(stream, state) { |
||||
state.combineTokens = null; |
||||
|
||||
// Hack to prevent formatting override inside code blocks (block and inline)
|
||||
if (state.codeBlock) { |
||||
if (stream.match(/^```+/)) { |
||||
state.codeBlock = false; |
||||
return null; |
||||
} |
||||
stream.skipToEnd(); |
||||
return null; |
||||
} |
||||
if (stream.sol()) { |
||||
state.code = false; |
||||
} |
||||
if (stream.sol() && stream.match(/^```+/)) { |
||||
stream.skipToEnd(); |
||||
state.codeBlock = true; |
||||
return null; |
||||
} |
||||
// If this block is changed, it may need to be updated in Markdown mode
|
||||
if (stream.peek() === '`') { |
||||
stream.next(); |
||||
var before = stream.pos; |
||||
stream.eatWhile('`'); |
||||
var difference = 1 + stream.pos - before; |
||||
if (!state.code) { |
||||
codeDepth = difference; |
||||
state.code = true; |
||||
} else { |
||||
if (difference === codeDepth) { // Must be exact
|
||||
state.code = false; |
||||
} |
||||
} |
||||
return null; |
||||
} else if (state.code) { |
||||
stream.next(); |
||||
return null; |
||||
} |
||||
// Check if space. If so, links can be formatted later on
|
||||
if (stream.eatSpace()) { |
||||
state.ateSpace = true; |
||||
return null; |
||||
} |
||||
if (stream.sol() || state.ateSpace) { |
||||
state.ateSpace = false; |
||||
if (modeConfig.gitHubSpice !== false) { |
||||
if(stream.match(/^(?:[a-zA-Z0-9\-_]+\/)?(?:[a-zA-Z0-9\-_]+@)?(?:[a-f0-9]{7,40}\b)/)) { |
||||
// User/Project@SHA
|
||||
// User@SHA
|
||||
// SHA
|
||||
state.combineTokens = true; |
||||
return "link"; |
||||
} else if (stream.match(/^(?:[a-zA-Z0-9\-_]+\/)?(?:[a-zA-Z0-9\-_]+)?#[0-9]+\b/)) { |
||||
// User/Project#Num
|
||||
// User#Num
|
||||
// #Num
|
||||
state.combineTokens = true; |
||||
return "link"; |
||||
} |
||||
} |
||||
} |
||||
if (stream.match(urlRE) && |
||||
stream.string.slice(stream.start - 2, stream.start) != "](" && |
||||
(stream.start == 0 || /\W/.test(stream.string.charAt(stream.start - 1)))) { |
||||
// URLs
|
||||
// Taken from http://daringfireball.net/2010/07/improved_regex_for_matching_urls
|
||||
// And then (issue #1160) simplified to make it not crash the Chrome Regexp engine
|
||||
// And then limited url schemes to the CommonMark list, so foo:bar isn't matched as a URL
|
||||
state.combineTokens = true; |
||||
return "link"; |
||||
} |
||||
stream.next(); |
||||
return null; |
||||
}, |
||||
blankLine: blankLine |
||||
}; |
||||
|
||||
var markdownConfig = { |
||||
underscoresBreakWords: false, |
||||
taskLists: true, |
||||
fencedCodeBlocks: '```', |
||||
strikethrough: true |
||||
}; |
||||
for (var attr in modeConfig) { |
||||
markdownConfig[attr] = modeConfig[attr]; |
||||
} |
||||
markdownConfig.name = "markdown"; |
||||
return CodeMirror.overlayMode(CodeMirror.getMode(config, markdownConfig), gfmOverlay); |
||||
|
||||
}, "markdown"); |
||||
|
||||
CodeMirror.defineMIME("text/x-gfm", "gfm"); |
||||
}); |
@ -0,0 +1,93 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: GFM mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/mode/overlay.js"></script> |
||||
<script src="../xml/xml.js"></script> |
||||
<script src="../markdown/markdown.js"></script> |
||||
<script src="gfm.js"></script> |
||||
<script src="../javascript/javascript.js"></script> |
||||
<script src="../css/css.js"></script> |
||||
<script src="../htmlmixed/htmlmixed.js"></script> |
||||
<script src="../clike/clike.js"></script> |
||||
<script src="../meta.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">GFM</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>GFM mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
GitHub Flavored Markdown |
||||
======================== |
||||
|
||||
Everything from markdown plus GFM features: |
||||
|
||||
## URL autolinking |
||||
|
||||
Underscores_are_allowed_between_words. |
||||
|
||||
## Strikethrough text |
||||
|
||||
GFM adds syntax to strikethrough text, which is missing from standard Markdown. |
||||
|
||||
~~Mistaken text.~~ |
||||
~~**works with other formatting**~~ |
||||
|
||||
~~spans across |
||||
lines~~ |
||||
|
||||
## Fenced code blocks (and syntax highlighting) |
||||
|
||||
```javascript |
||||
for (var i = 0; i < items.length; i++) { |
||||
console.log(items[i], i); // log them |
||||
} |
||||
``` |
||||
|
||||
## Task Lists |
||||
|
||||
- [ ] Incomplete task list item |
||||
- [x] **Completed** task list item |
||||
|
||||
## A bit of GitHub spice |
||||
|
||||
* SHA: be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 |
||||
* User@SHA ref: mojombo@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 |
||||
* User/Project@SHA: mojombo/god@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 |
||||
* \#Num: #1 |
||||
* User/#Num: mojombo#1 |
||||
* User/Project#Num: mojombo/god#1 |
||||
|
||||
See http://github.github.com/github-flavored-markdown/. |
||||
|
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
mode: 'gfm', |
||||
lineNumbers: true, |
||||
theme: "default" |
||||
}); |
||||
</script> |
||||
|
||||
<p>Optionally depends on other modes for properly highlighted code blocks.</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#gfm_*">normal</a>, <a href="../../test/index.html#verbose,gfm_*">verbose</a>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,236 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({tabSize: 4}, "gfm"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); } |
||||
var modeHighlightFormatting = CodeMirror.getMode({tabSize: 4}, {name: "gfm", highlightFormatting: true}); |
||||
function FT(name) { test.mode(name, modeHighlightFormatting, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
FT("codeBackticks", |
||||
"[comment&formatting&formatting-code `][comment foo][comment&formatting&formatting-code `]"); |
||||
|
||||
FT("doubleBackticks", |
||||
"[comment&formatting&formatting-code ``][comment foo ` bar][comment&formatting&formatting-code ``]"); |
||||
|
||||
FT("codeBlock", |
||||
"[comment&formatting&formatting-code-block ```css]", |
||||
"[tag foo]", |
||||
"[comment&formatting&formatting-code-block ```]"); |
||||
|
||||
FT("taskList", |
||||
"[variable-2&formatting&formatting-list&formatting-list-ul - ][meta&formatting&formatting-task [ ]]][variable-2 foo]", |
||||
"[variable-2&formatting&formatting-list&formatting-list-ul - ][property&formatting&formatting-task [x]]][variable-2 foo]"); |
||||
|
||||
FT("formatting_strikethrough", |
||||
"[strikethrough&formatting&formatting-strikethrough ~~][strikethrough foo][strikethrough&formatting&formatting-strikethrough ~~]"); |
||||
|
||||
FT("formatting_strikethrough", |
||||
"foo [strikethrough&formatting&formatting-strikethrough ~~][strikethrough bar][strikethrough&formatting&formatting-strikethrough ~~]"); |
||||
|
||||
MT("emInWordAsterisk", |
||||
"foo[em *bar*]hello"); |
||||
|
||||
MT("emInWordUnderscore", |
||||
"foo_bar_hello"); |
||||
|
||||
MT("emStrongUnderscore", |
||||
"[strong __][em&strong _foo__][em _] bar"); |
||||
|
||||
MT("fencedCodeBlocks", |
||||
"[comment ```]", |
||||
"[comment foo]", |
||||
"", |
||||
"[comment ```]", |
||||
"bar"); |
||||
|
||||
MT("fencedCodeBlockModeSwitching", |
||||
"[comment ```javascript]", |
||||
"[variable foo]", |
||||
"", |
||||
"[comment ```]", |
||||
"bar"); |
||||
|
||||
MT("fencedCodeBlockModeSwitchingObjc", |
||||
"[comment ```objective-c]", |
||||
"[keyword @property] [variable NSString] [operator *] [variable foo];", |
||||
"[comment ```]", |
||||
"bar"); |
||||
|
||||
MT("fencedCodeBlocksNoTildes", |
||||
"~~~", |
||||
"foo", |
||||
"~~~"); |
||||
|
||||
MT("taskListAsterisk", |
||||
"[variable-2 * []] foo]", // Invalid; must have space or x between []
|
||||
"[variable-2 * [ ]]bar]", // Invalid; must have space after ]
|
||||
"[variable-2 * [x]]hello]", // Invalid; must have space after ]
|
||||
"[variable-2 * ][meta [ ]]][variable-2 [world]]]", // Valid; tests reference style links
|
||||
" [variable-3 * ][property [x]]][variable-3 foo]"); // Valid; can be nested
|
||||
|
||||
MT("taskListPlus", |
||||
"[variable-2 + []] foo]", // Invalid; must have space or x between []
|
||||
"[variable-2 + [ ]]bar]", // Invalid; must have space after ]
|
||||
"[variable-2 + [x]]hello]", // Invalid; must have space after ]
|
||||
"[variable-2 + ][meta [ ]]][variable-2 [world]]]", // Valid; tests reference style links
|
||||
" [variable-3 + ][property [x]]][variable-3 foo]"); // Valid; can be nested
|
||||
|
||||
MT("taskListDash", |
||||
"[variable-2 - []] foo]", // Invalid; must have space or x between []
|
||||
"[variable-2 - [ ]]bar]", // Invalid; must have space after ]
|
||||
"[variable-2 - [x]]hello]", // Invalid; must have space after ]
|
||||
"[variable-2 - ][meta [ ]]][variable-2 [world]]]", // Valid; tests reference style links
|
||||
" [variable-3 - ][property [x]]][variable-3 foo]"); // Valid; can be nested
|
||||
|
||||
MT("taskListNumber", |
||||
"[variable-2 1. []] foo]", // Invalid; must have space or x between []
|
||||
"[variable-2 2. [ ]]bar]", // Invalid; must have space after ]
|
||||
"[variable-2 3. [x]]hello]", // Invalid; must have space after ]
|
||||
"[variable-2 4. ][meta [ ]]][variable-2 [world]]]", // Valid; tests reference style links
|
||||
" [variable-3 1. ][property [x]]][variable-3 foo]"); // Valid; can be nested
|
||||
|
||||
MT("SHA", |
||||
"foo [link be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2] bar"); |
||||
|
||||
MT("SHAEmphasis", |
||||
"[em *foo ][em&link be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2][em *]"); |
||||
|
||||
MT("shortSHA", |
||||
"foo [link be6a8cc] bar"); |
||||
|
||||
MT("tooShortSHA", |
||||
"foo be6a8c bar"); |
||||
|
||||
MT("longSHA", |
||||
"foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd22 bar"); |
||||
|
||||
MT("badSHA", |
||||
"foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cg2 bar"); |
||||
|
||||
MT("userSHA", |
||||
"foo [link bar@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2] hello"); |
||||
|
||||
MT("userSHAEmphasis", |
||||
"[em *foo ][em&link bar@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2][em *]"); |
||||
|
||||
MT("userProjectSHA", |
||||
"foo [link bar/hello@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2] world"); |
||||
|
||||
MT("userProjectSHAEmphasis", |
||||
"[em *foo ][em&link bar/hello@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2][em *]"); |
||||
|
||||
MT("num", |
||||
"foo [link #1] bar"); |
||||
|
||||
MT("numEmphasis", |
||||
"[em *foo ][em&link #1][em *]"); |
||||
|
||||
MT("badNum", |
||||
"foo #1bar hello"); |
||||
|
||||
MT("userNum", |
||||
"foo [link bar#1] hello"); |
||||
|
||||
MT("userNumEmphasis", |
||||
"[em *foo ][em&link bar#1][em *]"); |
||||
|
||||
MT("userProjectNum", |
||||
"foo [link bar/hello#1] world"); |
||||
|
||||
MT("userProjectNumEmphasis", |
||||
"[em *foo ][em&link bar/hello#1][em *]"); |
||||
|
||||
MT("vanillaLink", |
||||
"foo [link http://www.example.com/] bar"); |
||||
|
||||
MT("vanillaLinkNoScheme", |
||||
"foo [link www.example.com] bar"); |
||||
|
||||
MT("vanillaLinkHttps", |
||||
"foo [link https://www.example.com/] bar"); |
||||
|
||||
MT("vanillaLinkDataSchema", |
||||
"foo [link ] bar"); |
||||
|
||||
MT("vanillaLinkPunctuation", |
||||
"foo [link http://www.example.com/]. bar"); |
||||
|
||||
MT("vanillaLinkExtension", |
||||
"foo [link http://www.example.com/index.html] bar"); |
||||
|
||||
MT("vanillaLinkEmphasis", |
||||
"foo [em *][em&link http://www.example.com/index.html][em *] bar"); |
||||
|
||||
MT("notALink", |
||||
"foo asfd:asdf bar"); |
||||
|
||||
MT("notALink", |
||||
"[comment ```css]", |
||||
"[tag foo] {[property color]:[keyword black];}", |
||||
"[comment ```][link http://www.example.com/]"); |
||||
|
||||
MT("notALink", |
||||
"[comment ``foo `bar` http://www.example.com/``] hello"); |
||||
|
||||
MT("notALink", |
||||
"[comment `foo]", |
||||
"[comment&link http://www.example.com/]", |
||||
"[comment `] foo", |
||||
"", |
||||
"[link http://www.example.com/]"); |
||||
|
||||
MT("headerCodeBlockGithub", |
||||
"[header&header-1 # heading]", |
||||
"", |
||||
"[comment ```]", |
||||
"[comment code]", |
||||
"[comment ```]", |
||||
"", |
||||
"Commit: [link be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2]", |
||||
"Issue: [link #1]", |
||||
"Link: [link http://www.example.com/]"); |
||||
|
||||
MT("strikethrough", |
||||
"[strikethrough ~~foo~~]"); |
||||
|
||||
MT("strikethroughWithStartingSpace", |
||||
"~~ foo~~"); |
||||
|
||||
MT("strikethroughUnclosedStrayTildes", |
||||
"[strikethrough ~~foo~~~]"); |
||||
|
||||
MT("strikethroughUnclosedStrayTildes", |
||||
"[strikethrough ~~foo ~~]"); |
||||
|
||||
MT("strikethroughUnclosedStrayTildes", |
||||
"[strikethrough ~~foo ~~ bar]"); |
||||
|
||||
MT("strikethroughUnclosedStrayTildes", |
||||
"[strikethrough ~~foo ~~ bar~~]hello"); |
||||
|
||||
MT("strikethroughOneLetter", |
||||
"[strikethrough ~~a~~]"); |
||||
|
||||
MT("strikethroughWrapped", |
||||
"[strikethrough ~~foo]", |
||||
"[strikethrough foo~~]"); |
||||
|
||||
MT("strikethroughParagraph", |
||||
"[strikethrough ~~foo]", |
||||
"", |
||||
"foo[strikethrough ~~bar]"); |
||||
|
||||
MT("strikethroughEm", |
||||
"[strikethrough ~~foo][em&strikethrough *bar*][strikethrough ~~]"); |
||||
|
||||
MT("strikethroughEm", |
||||
"[em *][em&strikethrough ~~foo~~][em *]"); |
||||
|
||||
MT("strikethroughStrong", |
||||
"[strikethrough ~~][strong&strikethrough **foo**][strikethrough ~~]"); |
||||
|
||||
MT("strikethroughStrong", |
||||
"[strong **][strong&strikethrough ~~foo~~][strong **]"); |
||||
|
||||
})(); |
@ -0,0 +1,48 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Gherkin mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="gherkin.js"></script> |
||||
<style>.CodeMirror { border-top: 1px solid #ddd; border-bottom: 1px solid #ddd; }</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Gherkin</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Gherkin mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
Feature: Using Google |
||||
Background: |
||||
Something something |
||||
Something else |
||||
Scenario: Has a homepage |
||||
When I navigate to the google home page |
||||
Then the home page should contain the menu and the search form |
||||
Scenario: Searching for a term |
||||
When I navigate to the google home page |
||||
When I search for Tofu |
||||
Then the search results page is displayed |
||||
Then the search results page contains 10 individual search results |
||||
Then the search results contain a link to the wikipedia tofu page |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-feature</code>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,185 @@ |
||||
// 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("go", function(config) { |
||||
var indentUnit = config.indentUnit; |
||||
|
||||
var keywords = { |
||||
"break":true, "case":true, "chan":true, "const":true, "continue":true, |
||||
"default":true, "defer":true, "else":true, "fallthrough":true, "for":true, |
||||
"func":true, "go":true, "goto":true, "if":true, "import":true, |
||||
"interface":true, "map":true, "package":true, "range":true, "return":true, |
||||
"select":true, "struct":true, "switch":true, "type":true, "var":true, |
||||
"bool":true, "byte":true, "complex64":true, "complex128":true, |
||||
"float32":true, "float64":true, "int8":true, "int16":true, "int32":true, |
||||
"int64":true, "string":true, "uint8":true, "uint16":true, "uint32":true, |
||||
"uint64":true, "int":true, "uint":true, "uintptr":true, "error": true |
||||
}; |
||||
|
||||
var atoms = { |
||||
"true":true, "false":true, "iota":true, "nil":true, "append":true, |
||||
"cap":true, "close":true, "complex":true, "copy":true, "imag":true, |
||||
"len":true, "make":true, "new":true, "panic":true, "print":true, |
||||
"println":true, "real":true, "recover":true |
||||
}; |
||||
|
||||
var isOperatorChar = /[+\-*&^%:=<>!|\/]/; |
||||
|
||||
var curPunc; |
||||
|
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (ch == '"' || ch == "'" || ch == "`") { |
||||
state.tokenize = tokenString(ch); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
if (/[\d\.]/.test(ch)) { |
||||
if (ch == ".") { |
||||
stream.match(/^[0-9]+([eE][\-+]?[0-9]+)?/); |
||||
} else if (ch == "0") { |
||||
stream.match(/^[xX][0-9a-fA-F]+/) || stream.match(/^0[0-7]+/); |
||||
} else { |
||||
stream.match(/^[0-9]*\.?[0-9]*([eE][\-+]?[0-9]+)?/); |
||||
} |
||||
return "number"; |
||||
} |
||||
if (/[\[\]{}\(\),;\:\.]/.test(ch)) { |
||||
curPunc = ch; |
||||
return null; |
||||
} |
||||
if (ch == "/") { |
||||
if (stream.eat("*")) { |
||||
state.tokenize = tokenComment; |
||||
return tokenComment(stream, state); |
||||
} |
||||
if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_\xa1-\uffff]/); |
||||
var cur = stream.current(); |
||||
if (keywords.propertyIsEnumerable(cur)) { |
||||
if (cur == "case" || cur == "default") curPunc = "case"; |
||||
return "keyword"; |
||||
} |
||||
if (atoms.propertyIsEnumerable(cur)) return "atom"; |
||||
return "variable"; |
||||
} |
||||
|
||||
function tokenString(quote) { |
||||
return function(stream, state) { |
||||
var escaped = false, next, end = false; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == quote && !escaped) {end = true; break;} |
||||
escaped = !escaped && quote != "`" && next == "\\"; |
||||
} |
||||
if (end || !(escaped || quote == "`")) |
||||
state.tokenize = tokenBase; |
||||
return "string"; |
||||
}; |
||||
} |
||||
|
||||
function tokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = tokenBase; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return "comment"; |
||||
} |
||||
|
||||
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) { |
||||
if (!state.context.prev) return; |
||||
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: null, |
||||
context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), |
||||
indented: 0, |
||||
startOfLine: true |
||||
}; |
||||
}, |
||||
|
||||
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; |
||||
if (ctx.type == "case") ctx.type = "}"; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
curPunc = null; |
||||
var style = (state.tokenize || tokenBase)(stream, state); |
||||
if (style == "comment") return style; |
||||
if (ctx.align == null) ctx.align = true; |
||||
|
||||
if (curPunc == "{") pushContext(state, stream.column(), "}"); |
||||
else if (curPunc == "[") pushContext(state, stream.column(), "]"); |
||||
else if (curPunc == "(") pushContext(state, stream.column(), ")"); |
||||
else if (curPunc == "case") ctx.type = "case"; |
||||
else if (curPunc == "}" && ctx.type == "}") ctx = popContext(state); |
||||
else if (curPunc == ctx.type) popContext(state); |
||||
state.startOfLine = false; |
||||
return style; |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != tokenBase && state.tokenize != null) return 0; |
||||
var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); |
||||
if (ctx.type == "case" && /^(?:case|default)\b/.test(textAfter)) { |
||||
state.context.type = "}"; |
||||
return ctx.indented; |
||||
} |
||||
var closing = firstChar == ctx.type; |
||||
if (ctx.align) return ctx.column + (closing ? 0 : 1); |
||||
else return ctx.indented + (closing ? 0 : indentUnit); |
||||
}, |
||||
|
||||
electricChars: "{}):", |
||||
fold: "brace", |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/", |
||||
lineComment: "//" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-go", "go"); |
||||
|
||||
}); |
@ -0,0 +1,85 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Go mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/elegant.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="go.js"></script> |
||||
<style>.CodeMirror {border:1px solid #999; background:#ffc}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Go</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Go mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
// Prime Sieve in Go. |
||||
// Taken from the Go specification. |
||||
// Copyright © The Go Authors. |
||||
|
||||
package main |
||||
|
||||
import "fmt" |
||||
|
||||
// Send the sequence 2, 3, 4, ... to channel 'ch'. |
||||
func generate(ch chan<- int) { |
||||
for i := 2; ; i++ { |
||||
ch <- i // Send 'i' to channel 'ch' |
||||
} |
||||
} |
||||
|
||||
// Copy the values from channel 'src' to channel 'dst', |
||||
// removing those divisible by 'prime'. |
||||
func filter(src <-chan int, dst chan<- int, prime int) { |
||||
for i := range src { // Loop over values received from 'src'. |
||||
if i%prime != 0 { |
||||
dst <- i // Send 'i' to channel 'dst'. |
||||
} |
||||
} |
||||
} |
||||
|
||||
// The prime sieve: Daisy-chain filter processes together. |
||||
func sieve() { |
||||
ch := make(chan int) // Create a new channel. |
||||
go generate(ch) // Start generate() as a subprocess. |
||||
for { |
||||
prime := <-ch |
||||
fmt.Print(prime, "\n") |
||||
ch1 := make(chan int) |
||||
go filter(ch, ch1, prime) |
||||
ch = ch1 |
||||
} |
||||
} |
||||
|
||||
func main() { |
||||
sieve() |
||||
} |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
theme: "elegant", |
||||
matchBrackets: true, |
||||
indentUnit: 8, |
||||
tabSize: 8, |
||||
indentWithTabs: true, |
||||
mode: "text/x-go" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME type:</strong> <code>text/x-go</code></p> |
||||
</article> |
@ -0,0 +1,230 @@ |
||||
// 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("groovy", function(config) { |
||||
function words(str) { |
||||
var obj = {}, words = str.split(" "); |
||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; |
||||
return obj; |
||||
} |
||||
var keywords = words( |
||||
"abstract as assert boolean break byte case catch char class const continue def default " + |
||||
"do double else enum extends final finally float for goto if implements import in " + |
||||
"instanceof int interface long native new package private protected public return " + |
||||
"short static strictfp super switch synchronized threadsafe throw throws transient " + |
||||
"try void volatile while"); |
||||
var blockKeywords = words("catch class do else finally for if switch try while enum interface def"); |
||||
var standaloneKeywords = words("return break continue"); |
||||
var atoms = words("null true false this"); |
||||
|
||||
var curPunc; |
||||
function tokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (ch == '"' || ch == "'") { |
||||
return startString(ch, stream, state); |
||||
} |
||||
if (/[\[\]{}\(\),;\:\.]/.test(ch)) { |
||||
curPunc = ch; |
||||
return null; |
||||
} |
||||
if (/\d/.test(ch)) { |
||||
stream.eatWhile(/[\w\.]/); |
||||
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, false)) { |
||||
return startString(ch, stream, state); |
||||
} |
||||
} |
||||
if (ch == "-" && stream.eat(">")) { |
||||
curPunc = "->"; |
||||
return null; |
||||
} |
||||
if (/[+\-*&%=<>!?|\/~]/.test(ch)) { |
||||
stream.eatWhile(/[+\-*&%=<>|~]/); |
||||
return "operator"; |
||||
} |
||||
stream.eatWhile(/[\w\$_]/); |
||||
if (ch == "@") { stream.eatWhile(/[\w\$_\.]/); return "meta"; } |
||||
if (state.lastToken == ".") return "property"; |
||||
if (stream.eat(":")) { curPunc = "proplabel"; return "property"; } |
||||
var cur = stream.current(); |
||||
if (atoms.propertyIsEnumerable(cur)) { return "atom"; } |
||||
if (keywords.propertyIsEnumerable(cur)) { |
||||
if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; |
||||
else if (standaloneKeywords.propertyIsEnumerable(cur)) curPunc = "standalone"; |
||||
return "keyword"; |
||||
} |
||||
return "variable"; |
||||
} |
||||
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"; |
||||
} |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
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 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, newline) { |
||||
return !last || last == "operator" || last == "->" || /[\.\[\{\(,;:]/.test(last) || |
||||
last == "newstatement" || last == "keyword" || last == "proplabel" || |
||||
(last == "standalone" && !newline); |
||||
} |
||||
|
||||
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, true)) { |
||||
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); |
||||
// Handle indentation for {x -> \n ... }
|
||||
else if (curPunc == "->" && ctx.type == "statement" && ctx.prev.type == "}") { |
||||
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, true)) 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); |
||||
}, |
||||
|
||||
electricChars: "{}", |
||||
closeBrackets: {triples: "'\""}, |
||||
fold: "brace" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-groovy", "groovy"); |
||||
|
||||
}); |
@ -0,0 +1,84 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Groovy mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="groovy.js"></script> |
||||
<style>.CodeMirror {border-top: 1px solid #500; border-bottom: 1px solid #500;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Groovy</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Groovy mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
//Pattern for groovy script |
||||
def p = ~/.*\.groovy/ |
||||
new File( 'd:\\scripts' ).eachFileMatch(p) {f -> |
||||
// imports list |
||||
def imports = [] |
||||
f.eachLine { |
||||
// condition to detect an import instruction |
||||
ln -> if ( ln =~ '^import .*' ) { |
||||
imports << "${ln - 'import '}" |
||||
} |
||||
} |
||||
// print thmen |
||||
if ( ! imports.empty ) { |
||||
println f |
||||
imports.each{ println " $it" } |
||||
} |
||||
} |
||||
|
||||
/* Coin changer demo code from http://groovy.codehaus.org */ |
||||
|
||||
enum UsCoin { |
||||
quarter(25), dime(10), nickel(5), penny(1) |
||||
UsCoin(v) { value = v } |
||||
final value |
||||
} |
||||
|
||||
enum OzzieCoin { |
||||
fifty(50), twenty(20), ten(10), five(5) |
||||
OzzieCoin(v) { value = v } |
||||
final value |
||||
} |
||||
|
||||
def plural(word, count) { |
||||
if (count == 1) return word |
||||
word[-1] == 'y' ? word[0..-2] + "ies" : word + "s" |
||||
} |
||||
|
||||
def change(currency, amount) { |
||||
currency.values().inject([]){ list, coin -> |
||||
int count = amount / coin.value |
||||
amount = amount % coin.value |
||||
list += "$count ${plural(coin.toString(), count)}" |
||||
} |
||||
} |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: "text/x-groovy" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-groovy</code></p> |
||||
</article> |
@ -0,0 +1,161 @@ |
||||
// 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"), require("../htmlmixed/htmlmixed"), require("../ruby/ruby")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../htmlmixed/htmlmixed", "../ruby/ruby"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
// full haml mode. This handled embedded ruby and html fragments too
|
||||
CodeMirror.defineMode("haml", function(config) { |
||||
var htmlMode = CodeMirror.getMode(config, {name: "htmlmixed"}); |
||||
var rubyMode = CodeMirror.getMode(config, "ruby"); |
||||
|
||||
function rubyInQuote(endQuote) { |
||||
return function(stream, state) { |
||||
var ch = stream.peek(); |
||||
if (ch == endQuote && state.rubyState.tokenize.length == 1) { |
||||
// step out of ruby context as it seems to complete processing all the braces
|
||||
stream.next(); |
||||
state.tokenize = html; |
||||
return "closeAttributeTag"; |
||||
} else { |
||||
return ruby(stream, state); |
||||
} |
||||
}; |
||||
} |
||||
|
||||
function ruby(stream, state) { |
||||
if (stream.match("-#")) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
return rubyMode.token(stream, state.rubyState); |
||||
} |
||||
|
||||
function html(stream, state) { |
||||
var ch = stream.peek(); |
||||
|
||||
// handle haml declarations. All declarations that cant be handled here
|
||||
// will be passed to html mode
|
||||
if (state.previousToken.style == "comment" ) { |
||||
if (state.indented > state.previousToken.indented) { |
||||
stream.skipToEnd(); |
||||
return "commentLine"; |
||||
} |
||||
} |
||||
|
||||
if (state.startOfLine) { |
||||
if (ch == "!" && stream.match("!!")) { |
||||
stream.skipToEnd(); |
||||
return "tag"; |
||||
} else if (stream.match(/^%[\w:#\.]+=/)) { |
||||
state.tokenize = ruby; |
||||
return "hamlTag"; |
||||
} else if (stream.match(/^%[\w:]+/)) { |
||||
return "hamlTag"; |
||||
} else if (ch == "/" ) { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
|
||||
if (state.startOfLine || state.previousToken.style == "hamlTag") { |
||||
if ( ch == "#" || ch == ".") { |
||||
stream.match(/[\w-#\.]*/); |
||||
return "hamlAttribute"; |
||||
} |
||||
} |
||||
|
||||
// donot handle --> as valid ruby, make it HTML close comment instead
|
||||
if (state.startOfLine && !stream.match("-->", false) && (ch == "=" || ch == "-" )) { |
||||
state.tokenize = ruby; |
||||
return state.tokenize(stream, state); |
||||
} |
||||
|
||||
if (state.previousToken.style == "hamlTag" || |
||||
state.previousToken.style == "closeAttributeTag" || |
||||
state.previousToken.style == "hamlAttribute") { |
||||
if (ch == "(") { |
||||
state.tokenize = rubyInQuote(")"); |
||||
return state.tokenize(stream, state); |
||||
} else if (ch == "{") { |
||||
if (!stream.match(/^\{%.*/)) { |
||||
state.tokenize = rubyInQuote("}"); |
||||
return state.tokenize(stream, state); |
||||
} |
||||
} |
||||
} |
||||
|
||||
return htmlMode.token(stream, state.htmlState); |
||||
} |
||||
|
||||
return { |
||||
// default to html mode
|
||||
startState: function() { |
||||
var htmlState = CodeMirror.startState(htmlMode); |
||||
var rubyState = CodeMirror.startState(rubyMode); |
||||
return { |
||||
htmlState: htmlState, |
||||
rubyState: rubyState, |
||||
indented: 0, |
||||
previousToken: { style: null, indented: 0}, |
||||
tokenize: html |
||||
}; |
||||
}, |
||||
|
||||
copyState: function(state) { |
||||
return { |
||||
htmlState : CodeMirror.copyState(htmlMode, state.htmlState), |
||||
rubyState: CodeMirror.copyState(rubyMode, state.rubyState), |
||||
indented: state.indented, |
||||
previousToken: state.previousToken, |
||||
tokenize: state.tokenize |
||||
}; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (stream.sol()) { |
||||
state.indented = stream.indentation(); |
||||
state.startOfLine = true; |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
var style = state.tokenize(stream, state); |
||||
state.startOfLine = false; |
||||
// dont record comment line as we only want to measure comment line with
|
||||
// the opening comment block
|
||||
if (style && style != "commentLine") { |
||||
state.previousToken = { style: style, indented: state.indented }; |
||||
} |
||||
// if current state is ruby and the previous token is not `,` reset the
|
||||
// tokenize to html
|
||||
if (stream.eol() && state.tokenize == ruby) { |
||||
stream.backUp(1); |
||||
var ch = stream.peek(); |
||||
stream.next(); |
||||
if (ch && ch != ",") { |
||||
state.tokenize = html; |
||||
} |
||||
} |
||||
// reprocess some of the specific style tag when finish setting previousToken
|
||||
if (style == "hamlTag") { |
||||
style = "tag"; |
||||
} else if (style == "commentLine") { |
||||
style = "comment"; |
||||
} else if (style == "hamlAttribute") { |
||||
style = "attribute"; |
||||
} else if (style == "closeAttributeTag") { |
||||
style = null; |
||||
} |
||||
return style; |
||||
} |
||||
}; |
||||
}, "htmlmixed", "ruby"); |
||||
|
||||
CodeMirror.defineMIME("text/x-haml", "haml"); |
||||
}); |
@ -0,0 +1,79 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: HAML mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../xml/xml.js"></script> |
||||
<script src="../htmlmixed/htmlmixed.js"></script> |
||||
<script src="../javascript/javascript.js"></script> |
||||
<script src="../ruby/ruby.js"></script> |
||||
<script src="haml.js"></script> |
||||
<style>.CodeMirror {background: #f8f8f8;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">HAML</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>HAML mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
!!! |
||||
#content |
||||
.left.column(title="title"){:href => "/hello", :test => "#{hello}_#{world}"} |
||||
<!-- This is a comment --> |
||||
%h2 Welcome to our site! |
||||
%p= puts "HAML MODE" |
||||
.right.column |
||||
= render :partial => "sidebar" |
||||
|
||||
.container |
||||
.row |
||||
.span8 |
||||
%h1.title= @page_title |
||||
%p.title= @page_title |
||||
%p |
||||
/ |
||||
The same as HTML comment |
||||
Hello multiline comment |
||||
|
||||
-# haml comment |
||||
This wont be displayed |
||||
nor will this |
||||
Date/Time: |
||||
- now = DateTime.now |
||||
%strong= now |
||||
- if now > DateTime.parse("December 31, 2006") |
||||
= "Happy new " + "year!" |
||||
|
||||
%title |
||||
= @title |
||||
\= @title |
||||
<h1>Title</h1> |
||||
<h1 title="HELLO"> |
||||
Title |
||||
</h1> |
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
mode: "text/x-haml" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-haml</code>.</p> |
||||
|
||||
<p><strong>Parsing/Highlighting Tests:</strong> <a href="../../test/index.html#haml_*">normal</a>, <a href="../../test/index.html#verbose,haml_*">verbose</a>.</p> |
||||
|
||||
</article> |
@ -0,0 +1,97 @@ |
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
||||
|
||||
(function() { |
||||
var mode = CodeMirror.getMode({tabSize: 4, indentUnit: 2}, "haml"); |
||||
function MT(name) { test.mode(name, mode, Array.prototype.slice.call(arguments, 1)); } |
||||
|
||||
// Requires at least one media query
|
||||
MT("elementName", |
||||
"[tag %h1] Hey There"); |
||||
|
||||
MT("oneElementPerLine", |
||||
"[tag %h1] Hey There %h2"); |
||||
|
||||
MT("idSelector", |
||||
"[tag %h1][attribute #test] Hey There"); |
||||
|
||||
MT("classSelector", |
||||
"[tag %h1][attribute .hello] Hey There"); |
||||
|
||||
MT("docType", |
||||
"[tag !!! XML]"); |
||||
|
||||
MT("comment", |
||||
"[comment / Hello WORLD]"); |
||||
|
||||
MT("notComment", |
||||
"[tag %h1] This is not a / comment "); |
||||
|
||||
MT("attributes", |
||||
"[tag %a]([variable title][operator =][string \"test\"]){[atom :title] [operator =>] [string \"test\"]}"); |
||||
|
||||
MT("htmlCode", |
||||
"[tag&bracket <][tag h1][tag&bracket >]Title[tag&bracket </][tag h1][tag&bracket >]"); |
||||
|
||||
MT("rubyBlock", |
||||
"[operator =][variable-2 @item]"); |
||||
|
||||
MT("selectorRubyBlock", |
||||
"[tag %a.selector=] [variable-2 @item]"); |
||||
|
||||
MT("nestedRubyBlock", |
||||
"[tag %a]", |
||||
" [operator =][variable puts] [string \"test\"]"); |
||||
|
||||
MT("multilinePlaintext", |
||||
"[tag %p]", |
||||
" Hello,", |
||||
" World"); |
||||
|
||||
MT("multilineRuby", |
||||
"[tag %p]", |
||||
" [comment -# this is a comment]", |
||||
" [comment and this is a comment too]", |
||||
" Date/Time", |
||||
" [operator -] [variable now] [operator =] [tag DateTime][operator .][property now]", |
||||
" [tag %strong=] [variable now]", |
||||
" [operator -] [keyword if] [variable now] [operator >] [tag DateTime][operator .][property parse]([string \"December 31, 2006\"])", |
||||
" [operator =][string \"Happy\"]", |
||||
" [operator =][string \"Belated\"]", |
||||
" [operator =][string \"Birthday\"]"); |
||||
|
||||
MT("multilineComment", |
||||
"[comment /]", |
||||
" [comment Multiline]", |
||||
" [comment Comment]"); |
||||
|
||||
MT("hamlComment", |
||||
"[comment -# this is a comment]"); |
||||
|
||||
MT("multilineHamlComment", |
||||
"[comment -# this is a comment]", |
||||
" [comment and this is a comment too]"); |
||||
|
||||
MT("multilineHTMLComment", |
||||
"[comment <!--]", |
||||
" [comment what a comment]", |
||||
" [comment -->]"); |
||||
|
||||
MT("hamlAfterRubyTag", |
||||
"[attribute .block]", |
||||
" [tag %strong=] [variable now]", |
||||
" [attribute .test]", |
||||
" [operator =][variable now]", |
||||
" [attribute .right]"); |
||||
|
||||
MT("stretchedRuby", |
||||
"[operator =] [variable puts] [string \"Hello\"],", |
||||
" [string \"World\"]"); |
||||
|
||||
MT("interpolationInHashAttribute", |
||||
//"[tag %div]{[atom :id] [operator =>] [string \"#{][variable test][string }_#{][variable ting][string }\"]} test");
|
||||
"[tag %div]{[atom :id] [operator =>] [string \"#{][variable test][string }_#{][variable ting][string }\"]} test"); |
||||
|
||||
MT("interpolationInHTMLAttribute", |
||||
"[tag %div]([variable title][operator =][string \"#{][variable test][string }_#{][variable ting]()[string }\"]) Test"); |
||||
})(); |
@ -0,0 +1,62 @@ |
||||
// 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"), require("../../addon/mode/simple"), require("../../addon/mode/multiplex")); |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../../addon/mode/simple", "../../addon/mode/multiplex"], mod); |
||||
else // Plain browser env
|
||||
mod(CodeMirror); |
||||
})(function(CodeMirror) { |
||||
"use strict"; |
||||
|
||||
CodeMirror.defineSimpleMode("handlebars-tags", { |
||||
start: [ |
||||
{ regex: /\{\{!--/, push: "dash_comment", token: "comment" }, |
||||
{ regex: /\{\{!/, push: "comment", token: "comment" }, |
||||
{ regex: /\{\{/, push: "handlebars", token: "tag" } |
||||
], |
||||
handlebars: [ |
||||
{ regex: /\}\}/, pop: true, token: "tag" }, |
||||
|
||||
// Double and single quotes
|
||||
{ regex: /"(?:[^\\"]|\\.)*"?/, token: "string" }, |
||||
{ regex: /'(?:[^\\']|\\.)*'?/, token: "string" }, |
||||
|
||||
// Handlebars keywords
|
||||
{ regex: />|[#\/]([A-Za-z_]\w*)/, token: "keyword" }, |
||||
{ regex: /(?:else|this)\b/, token: "keyword" }, |
||||
|
||||
// Numeral
|
||||
{ regex: /\d+/i, token: "number" }, |
||||
|
||||
// Atoms like = and .
|
||||
{ regex: /=|~|@|true|false/, token: "atom" }, |
||||
|
||||
// Paths
|
||||
{ regex: /(?:\.\.\/)*(?:[A-Za-z_][\w\.]*)+/, token: "variable-2" } |
||||
], |
||||
dash_comment: [ |
||||
{ regex: /--\}\}/, pop: true, token: "comment" }, |
||||
|
||||
// Commented code
|
||||
{ regex: /./, token: "comment"} |
||||
], |
||||
comment: [ |
||||
{ regex: /\}\}/, pop: true, token: "comment" }, |
||||
{ regex: /./, token: "comment" } |
||||
] |
||||
}); |
||||
|
||||
CodeMirror.defineMode("handlebars", function(config, parserConfig) { |
||||
var handlebars = CodeMirror.getMode(config, "handlebars-tags"); |
||||
if (!parserConfig || !parserConfig.base) return handlebars; |
||||
return CodeMirror.multiplexingMode( |
||||
CodeMirror.getMode(config, parserConfig.base), |
||||
{open: "{{", close: "}}", mode: handlebars, parseDelimiters: true} |
||||
); |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-handlebars-template", "handlebars"); |
||||
}); |
@ -0,0 +1,79 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Handlebars mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/mode/simple.js"></script> |
||||
<script src="../../addon/mode/multiplex.js"></script> |
||||
<script src="../xml/xml.js"></script> |
||||
<script src="handlebars.js"></script> |
||||
<style>.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">HTML mixed</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Handlebars</h2> |
||||
<form><textarea id="code" name="code"> |
||||
{{> breadcrumbs}} |
||||
|
||||
{{!-- |
||||
You can use the t function to get |
||||
content translated to the current locale, es: |
||||
{{t 'article_list'}} |
||||
--}} |
||||
|
||||
<h1>{{t 'article_list'}}</h1> |
||||
|
||||
{{! one line comment }} |
||||
|
||||
{{#each articles}} |
||||
{{~title}} |
||||
<p>{{excerpt body size=120 ellipsis=true}}</p> |
||||
|
||||
{{#with author}} |
||||
written by {{first_name}} {{last_name}} |
||||
from category: {{../category.title}} |
||||
{{#if @../last}}foobar!{{/if}} |
||||
{{/with~}} |
||||
|
||||
{{#if promoted.latest}}Read this one! {{else}} This is ok! {{/if}} |
||||
|
||||
{{#if @last}}<hr>{{/if}} |
||||
{{/each}} |
||||
|
||||
{{#form new_comment}} |
||||
<input type="text" name="body"> |
||||
{{/form}} |
||||
|
||||
</textarea></form> |
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
mode: {name: "handlebars", base: "text/html"} |
||||
}); |
||||
</script> |
||||
|
||||
<p>Handlebars syntax highlighting for CodeMirror.</p> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-handlebars-template</code></p> |
||||
|
||||
<p>Supported options: <code>base</code> to set the mode to |
||||
wrap. For example, use</p> |
||||
<pre>mode: {name: "handlebars", base: "text/html"}</pre> |
||||
<p>to highlight an HTML template.</p> |
||||
</article> |
@ -0,0 +1,43 @@ |
||||
// 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"), require("../haskell/haskell")) |
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["../../lib/codemirror", "../haskell/haskell"], mod) |
||||
else // Plain browser env
|
||||
mod(CodeMirror) |
||||
})(function (CodeMirror) { |
||||
"use strict" |
||||
|
||||
CodeMirror.defineMode("haskell-literate", function (config, parserConfig) { |
||||
var baseMode = CodeMirror.getMode(config, (parserConfig && parserConfig.base) || "haskell") |
||||
|
||||
return { |
||||
startState: function () { |
||||
return { |
||||
inCode: false, |
||||
baseState: CodeMirror.startState(baseMode) |
||||
} |
||||
}, |
||||
token: function (stream, state) { |
||||
if (stream.sol()) { |
||||
if (state.inCode = stream.eat(">")) |
||||
return "meta" |
||||
} |
||||
if (state.inCode) { |
||||
return baseMode.token(stream, state.baseState) |
||||
} else { |
||||
stream.skipToEnd() |
||||
return "comment" |
||||
} |
||||
}, |
||||
innerMode: function (state) { |
||||
return state.inCode ? {state: state.baseState, mode: baseMode} : null |
||||
} |
||||
} |
||||
}, "haskell") |
||||
|
||||
CodeMirror.defineMIME("text/x-literate-haskell", "haskell-literate") |
||||
}); |
@ -0,0 +1,282 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Haskell-literate mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="haskell-literate.js"></script> |
||||
<script src="../haskell/haskell.js"></script> |
||||
<style>.CodeMirror { |
||||
border-top : 1px solid #DDDDDD; |
||||
border-bottom : 1px solid #DDDDDD; |
||||
}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo |
||||
src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Haskell-literate</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Haskell literate mode</h2> |
||||
<form> |
||||
<textarea id="code" name="code"> |
||||
> {-# LANGUAGE OverloadedStrings #-} |
||||
> {-# OPTIONS_GHC -fno-warn-unused-do-bind #-} |
||||
> import Control.Applicative ((<$>), (<*>)) |
||||
> import Data.Maybe (isJust) |
||||
|
||||
> import Data.Text (Text) |
||||
> import Text.Blaze ((!)) |
||||
> import qualified Data.Text as T |
||||
> import qualified Happstack.Server as Happstack |
||||
> import qualified Text.Blaze.Html5 as H |
||||
> import qualified Text.Blaze.Html5.Attributes as A |
||||
|
||||
> import Text.Digestive |
||||
> import Text.Digestive.Blaze.Html5 |
||||
> import Text.Digestive.Happstack |
||||
> import Text.Digestive.Util |
||||
|
||||
Simple forms and validation |
||||
--------------------------- |
||||
|
||||
Let's start by creating a very simple datatype to represent a user: |
||||
|
||||
> data User = User |
||||
> { userName :: Text |
||||
> , userMail :: Text |
||||
> } deriving (Show) |
||||
|
||||
And dive in immediately to create a `Form` for a user. The `Form v m a` type |
||||
has three parameters: |
||||
|
||||
- `v`: the type for messages and errors (usually a `String`-like type, `Text` in |
||||
this case); |
||||
- `m`: the monad we are operating in, not specified here; |
||||
- `a`: the return type of the `Form`, in this case, this is obviously `User`. |
||||
|
||||
> userForm :: Monad m => Form Text m User |
||||
|
||||
We create forms by using the `Applicative` interface. A few form types are |
||||
provided in the `Text.Digestive.Form` module, such as `text`, `string`, |
||||
`bool`... |
||||
|
||||
In the `digestive-functors` library, the developer is required to label each |
||||
field using the `.:` operator. This might look like a bit of a burden, but it |
||||
allows you to do some really useful stuff, like separating the `Form` from the |
||||
actual HTML layout. |
||||
|
||||
> userForm = User |
||||
> <$> "name" .: text Nothing |
||||
> <*> "mail" .: check "Not a valid email address" checkEmail (text Nothing) |
||||
|
||||
The `check` function enables you to validate the result of a form. For example, |
||||
we can validate the email address with a really naive `checkEmail` function. |
||||
|
||||
> checkEmail :: Text -> Bool |
||||
> checkEmail = isJust . T.find (== '@') |
||||
|
||||
More validation |
||||
--------------- |
||||
|
||||
For our example, we also want descriptions of Haskell libraries, and in order to |
||||
do that, we need package versions... |
||||
|
||||
> type Version = [Int] |
||||
|
||||
We want to let the user input a version number such as `0.1.0.0`. This means we |
||||
need to validate if the input `Text` is of this form, and then we need to parse |
||||
it to a `Version` type. Fortunately, we can do this in a single function: |
||||
`validate` allows conversion between values, which can optionally fail. |
||||
|
||||
`readMaybe :: Read a => String -> Maybe a` is a utility function imported from |
||||
`Text.Digestive.Util`. |
||||
|
||||
> validateVersion :: Text -> Result Text Version |
||||
> validateVersion = maybe (Error "Cannot parse version") Success . |
||||
> mapM (readMaybe . T.unpack) . T.split (== '.') |
||||
|
||||
A quick test in GHCi: |
||||
|
||||
ghci> validateVersion (T.pack "0.3.2.1") |
||||
Success [0,3,2,1] |
||||
ghci> validateVersion (T.pack "0.oops") |
||||
Error "Cannot parse version" |
||||
|
||||
It works! This means we can now easily add a `Package` type and a `Form` for it: |
||||
|
||||
> data Category = Web | Text | Math |
||||
> deriving (Bounded, Enum, Eq, Show) |
||||
|
||||
> data Package = Package Text Version Category |
||||
> deriving (Show) |
||||
|
||||
> packageForm :: Monad m => Form Text m Package |
||||
> packageForm = Package |
||||
> <$> "name" .: text Nothing |
||||
> <*> "version" .: validate validateVersion (text (Just "0.0.0.1")) |
||||
> <*> "category" .: choice categories Nothing |
||||
> where |
||||
> categories = [(x, T.pack (show x)) | x <- [minBound .. maxBound]] |
||||
|
||||
Composing forms |
||||
--------------- |
||||
|
||||
A release has an author and a package. Let's use this to illustrate the |
||||
composability of the digestive-functors library: we can reuse the forms we have |
||||
written earlier on. |
||||
|
||||
> data Release = Release User Package |
||||
> deriving (Show) |
||||
|
||||
> releaseForm :: Monad m => Form Text m Release |
||||
> releaseForm = Release |
||||
> <$> "author" .: userForm |
||||
> <*> "package" .: packageForm |
||||
|
||||
Views |
||||
----- |
||||
|
||||
As mentioned before, one of the advantages of using digestive-functors is |
||||
separation of forms and their actual HTML layout. In order to do this, we have |
||||
another type, `View`. |
||||
|
||||
We can get a `View` from a `Form` by supplying input. A `View` contains more |
||||
information than a `Form`, it has: |
||||
|
||||
- the original form; |
||||
- the input given by the user; |
||||
- any errors that have occurred. |
||||
|
||||
It is this view that we convert to HTML. For this tutorial, we use the |
||||
[blaze-html] library, and some helpers from the `digestive-functors-blaze` |
||||
library. |
||||
|
||||
[blaze-html]: http://jaspervdj.be/blaze/ |
||||
|
||||
Let's write a view for the `User` form. As you can see, we here refer to the |
||||
different fields in the `userForm`. The `errorList` will generate a list of |
||||
errors for the `"mail"` field. |
||||
|
||||
> userView :: View H.Html -> H.Html |
||||
> userView view = do |
||||
> label "name" view "Name: " |
||||
> inputText "name" view |
||||
> H.br |
||||
> |
||||
> errorList "mail" view |
||||
> label "mail" view "Email address: " |
||||
> inputText "mail" view |
||||
> H.br |
||||
|
||||
Like forms, views are also composable: let's illustrate that by adding a view |
||||
for the `releaseForm`, in which we reuse `userView`. In order to do this, we |
||||
take only the parts relevant to the author from the view by using `subView`. We |
||||
can then pass the resulting view to our own `userView`. |
||||
We have no special view code for `Package`, so we can just add that to |
||||
`releaseView` as well. `childErrorList` will generate a list of errors for each |
||||
child of the specified form. In this case, this means a list of errors from |
||||
`"package.name"` and `"package.version"`. Note how we use `foo.bar` to refer to |
||||
nested forms. |
||||
|
||||
> releaseView :: View H.Html -> H.Html |
||||
> releaseView view = do |
||||
> H.h2 "Author" |
||||
> userView $ subView "author" view |
||||
> |
||||
> H.h2 "Package" |
||||
> childErrorList "package" view |
||||
> |
||||
> label "package.name" view "Name: " |
||||
> inputText "package.name" view |
||||
> H.br |
||||
> |
||||
> label "package.version" view "Version: " |
||||
> inputText "package.version" view |
||||
> H.br |
||||
> |
||||
> label "package.category" view "Category: " |
||||
> inputSelect "package.category" view |
||||
> H.br |
||||
|
||||
The attentive reader might have wondered what the type parameter for `View` is: |
||||
it is the `String`-like type used for e.g. error messages. |
||||
But wait! We have |
||||
releaseForm :: Monad m => Form Text m Release |
||||
releaseView :: View H.Html -> H.Html |
||||
... doesn't this mean that we need a `View Text` rather than a `View Html`? The |
||||
answer is yes -- but having `View Html` allows us to write these views more |
||||
easily with the `digestive-functors-blaze` library. Fortunately, we will be able |
||||
to fix this using the `Functor` instance of `View`. |
||||
fmap :: Monad m => (v -> w) -> View v -> View w |
||||
A backend |
||||
--------- |
||||
To finish this tutorial, we need to be able to actually run this code. We need |
||||
an HTTP server for that, and we use [Happstack] for this tutorial. The |
||||
`digestive-functors-happstack` library gives about everything we need for this. |
||||
[Happstack]: http://happstack.com/ |
||||
|
||||
> site :: Happstack.ServerPart Happstack.Response |
||||
> site = do |
||||
> Happstack.decodeBody $ Happstack.defaultBodyPolicy "/tmp" 4096 4096 4096 |
||||
> r <- runForm "test" releaseForm |
||||
> case r of |
||||
> (view, Nothing) -> do |
||||
> let view' = fmap H.toHtml view |
||||
> Happstack.ok $ Happstack.toResponse $ |
||||
> template $ |
||||
> form view' "/" $ do |
||||
> releaseView view' |
||||
> H.br |
||||
> inputSubmit "Submit" |
||||
> (_, Just release) -> Happstack.ok $ Happstack.toResponse $ |
||||
> template $ do |
||||
> css |
||||
> H.h1 "Release received" |
||||
> H.p $ H.toHtml $ show release |
||||
> |
||||
> main :: IO () |
||||
> main = Happstack.simpleHTTP Happstack.nullConf site |
||||
|
||||
Utilities |
||||
--------- |
||||
|
||||
> template :: H.Html -> H.Html |
||||
> template body = H.docTypeHtml $ do |
||||
> H.head $ do |
||||
> H.title "digestive-functors tutorial" |
||||
> css |
||||
> H.body body |
||||
> css :: H.Html |
||||
> css = H.style ! A.type_ "text/css" $ do |
||||
> "label {width: 130px; float: left; clear: both}" |
||||
> "ul.digestive-functors-error-list {" |
||||
> " color: red;" |
||||
> " list-style-type: none;" |
||||
> " padding-left: 0px;" |
||||
> "}" |
||||
</textarea> |
||||
</form> |
||||
|
||||
<p><strong>MIME types |
||||
defined:</strong> <code>text/x-literate-haskell</code>.</p> |
||||
|
||||
<p>Parser configuration parameters recognized: <code>base</code> to |
||||
set the base mode (defaults to <code>"haskell"</code>).</p> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {mode: "haskell-literate"}); |
||||
</script> |
||||
|
||||
</article> |
@ -0,0 +1,267 @@ |
||||
// 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("haskell", function(_config, modeConfig) { |
||||
|
||||
function switchState(source, setState, f) { |
||||
setState(f); |
||||
return f(source, setState); |
||||
} |
||||
|
||||
// These should all be Unicode extended, as per the Haskell 2010 report
|
||||
var smallRE = /[a-z_]/; |
||||
var largeRE = /[A-Z]/; |
||||
var digitRE = /\d/; |
||||
var hexitRE = /[0-9A-Fa-f]/; |
||||
var octitRE = /[0-7]/; |
||||
var idRE = /[a-z_A-Z0-9'\xa1-\uffff]/; |
||||
var symbolRE = /[-!#$%&*+.\/<=>?@\\^|~:]/; |
||||
var specialRE = /[(),;[\]`{}]/; |
||||
var whiteCharRE = /[ \t\v\f]/; // newlines are handled in tokenizer
|
||||
|
||||
function normal(source, setState) { |
||||
if (source.eatWhile(whiteCharRE)) { |
||||
return null; |
||||
} |
||||
|
||||
var ch = source.next(); |
||||
if (specialRE.test(ch)) { |
||||
if (ch == '{' && source.eat('-')) { |
||||
var t = "comment"; |
||||
if (source.eat('#')) { |
||||
t = "meta"; |
||||
} |
||||
return switchState(source, setState, ncomment(t, 1)); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
if (ch == '\'') { |
||||
if (source.eat('\\')) { |
||||
source.next(); // should handle other escapes here
|
||||
} |
||||
else { |
||||
source.next(); |
||||
} |
||||
if (source.eat('\'')) { |
||||
return "string"; |
||||
} |
||||
return "error"; |
||||
} |
||||
|
||||
if (ch == '"') { |
||||
return switchState(source, setState, stringLiteral); |
||||
} |
||||
|
||||
if (largeRE.test(ch)) { |
||||
source.eatWhile(idRE); |
||||
if (source.eat('.')) { |
||||
return "qualifier"; |
||||
} |
||||
return "variable-2"; |
||||
} |
||||
|
||||
if (smallRE.test(ch)) { |
||||
source.eatWhile(idRE); |
||||
return "variable"; |
||||
} |
||||
|
||||
if (digitRE.test(ch)) { |
||||
if (ch == '0') { |
||||
if (source.eat(/[xX]/)) { |
||||
source.eatWhile(hexitRE); // should require at least 1
|
||||
return "integer"; |
||||
} |
||||
if (source.eat(/[oO]/)) { |
||||
source.eatWhile(octitRE); // should require at least 1
|
||||
return "number"; |
||||
} |
||||
} |
||||
source.eatWhile(digitRE); |
||||
var t = "number"; |
||||
if (source.match(/^\.\d+/)) { |
||||
t = "number"; |
||||
} |
||||
if (source.eat(/[eE]/)) { |
||||
t = "number"; |
||||
source.eat(/[-+]/); |
||||
source.eatWhile(digitRE); // should require at least 1
|
||||
} |
||||
return t; |
||||
} |
||||
|
||||
if (ch == "." && source.eat(".")) |
||||
return "keyword"; |
||||
|
||||
if (symbolRE.test(ch)) { |
||||
if (ch == '-' && source.eat(/-/)) { |
||||
source.eatWhile(/-/); |
||||
if (!source.eat(symbolRE)) { |
||||
source.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
} |
||||
var t = "variable"; |
||||
if (ch == ':') { |
||||
t = "variable-2"; |
||||
} |
||||
source.eatWhile(symbolRE); |
||||
return t; |
||||
} |
||||
|
||||
return "error"; |
||||
} |
||||
|
||||
function ncomment(type, nest) { |
||||
if (nest == 0) { |
||||
return normal; |
||||
} |
||||
return function(source, setState) { |
||||
var currNest = nest; |
||||
while (!source.eol()) { |
||||
var ch = source.next(); |
||||
if (ch == '{' && source.eat('-')) { |
||||
++currNest; |
||||
} |
||||
else if (ch == '-' && source.eat('}')) { |
||||
--currNest; |
||||
if (currNest == 0) { |
||||
setState(normal); |
||||
return type; |
||||
} |
||||
} |
||||
} |
||||
setState(ncomment(type, currNest)); |
||||
return type; |
||||
}; |
||||
} |
||||
|
||||
function stringLiteral(source, setState) { |
||||
while (!source.eol()) { |
||||
var ch = source.next(); |
||||
if (ch == '"') { |
||||
setState(normal); |
||||
return "string"; |
||||
} |
||||
if (ch == '\\') { |
||||
if (source.eol() || source.eat(whiteCharRE)) { |
||||
setState(stringGap); |
||||
return "string"; |
||||
} |
||||
if (source.eat('&')) { |
||||
} |
||||
else { |
||||
source.next(); // should handle other escapes here
|
||||
} |
||||
} |
||||
} |
||||
setState(normal); |
||||
return "error"; |
||||
} |
||||
|
||||
function stringGap(source, setState) { |
||||
if (source.eat('\\')) { |
||||
return switchState(source, setState, stringLiteral); |
||||
} |
||||
source.next(); |
||||
setState(normal); |
||||
return "error"; |
||||
} |
||||
|
||||
|
||||
var wellKnownWords = (function() { |
||||
var wkw = {}; |
||||
function setType(t) { |
||||
return function () { |
||||
for (var i = 0; i < arguments.length; i++) |
||||
wkw[arguments[i]] = t; |
||||
}; |
||||
} |
||||
|
||||
setType("keyword")( |
||||
"case", "class", "data", "default", "deriving", "do", "else", "foreign", |
||||
"if", "import", "in", "infix", "infixl", "infixr", "instance", "let", |
||||
"module", "newtype", "of", "then", "type", "where", "_"); |
||||
|
||||
setType("keyword")( |
||||
"\.\.", ":", "::", "=", "\\", "\"", "<-", "->", "@", "~", "=>"); |
||||
|
||||
setType("builtin")( |
||||
"!!", "$!", "$", "&&", "+", "++", "-", ".", "/", "/=", "<", "<=", "=<<", |
||||
"==", ">", ">=", ">>", ">>=", "^", "^^", "||", "*", "**"); |
||||
|
||||
setType("builtin")( |
||||
"Bool", "Bounded", "Char", "Double", "EQ", "Either", "Enum", "Eq", |
||||
"False", "FilePath", "Float", "Floating", "Fractional", "Functor", "GT", |
||||
"IO", "IOError", "Int", "Integer", "Integral", "Just", "LT", "Left", |
||||
"Maybe", "Monad", "Nothing", "Num", "Ord", "Ordering", "Rational", "Read", |
||||
"ReadS", "Real", "RealFloat", "RealFrac", "Right", "Show", "ShowS", |
||||
"String", "True"); |
||||
|
||||
setType("builtin")( |
||||
"abs", "acos", "acosh", "all", "and", "any", "appendFile", "asTypeOf", |
||||
"asin", "asinh", "atan", "atan2", "atanh", "break", "catch", "ceiling", |
||||
"compare", "concat", "concatMap", "const", "cos", "cosh", "curry", |
||||
"cycle", "decodeFloat", "div", "divMod", "drop", "dropWhile", "either", |
||||
"elem", "encodeFloat", "enumFrom", "enumFromThen", "enumFromThenTo", |
||||
"enumFromTo", "error", "even", "exp", "exponent", "fail", "filter", |
||||
"flip", "floatDigits", "floatRadix", "floatRange", "floor", "fmap", |
||||
"foldl", "foldl1", "foldr", "foldr1", "fromEnum", "fromInteger", |
||||
"fromIntegral", "fromRational", "fst", "gcd", "getChar", "getContents", |
||||
"getLine", "head", "id", "init", "interact", "ioError", "isDenormalized", |
||||
"isIEEE", "isInfinite", "isNaN", "isNegativeZero", "iterate", "last", |
||||
"lcm", "length", "lex", "lines", "log", "logBase", "lookup", "map", |
||||
"mapM", "mapM_", "max", "maxBound", "maximum", "maybe", "min", "minBound", |
||||
"minimum", "mod", "negate", "not", "notElem", "null", "odd", "or", |
||||
"otherwise", "pi", "pred", "print", "product", "properFraction", |
||||
"putChar", "putStr", "putStrLn", "quot", "quotRem", "read", "readFile", |
||||
"readIO", "readList", "readLn", "readParen", "reads", "readsPrec", |
||||
"realToFrac", "recip", "rem", "repeat", "replicate", "return", "reverse", |
||||
"round", "scaleFloat", "scanl", "scanl1", "scanr", "scanr1", "seq", |
||||
"sequence", "sequence_", "show", "showChar", "showList", "showParen", |
||||
"showString", "shows", "showsPrec", "significand", "signum", "sin", |
||||
"sinh", "snd", "span", "splitAt", "sqrt", "subtract", "succ", "sum", |
||||
"tail", "take", "takeWhile", "tan", "tanh", "toEnum", "toInteger", |
||||
"toRational", "truncate", "uncurry", "undefined", "unlines", "until", |
||||
"unwords", "unzip", "unzip3", "userError", "words", "writeFile", "zip", |
||||
"zip3", "zipWith", "zipWith3"); |
||||
|
||||
var override = modeConfig.overrideKeywords; |
||||
if (override) for (var word in override) if (override.hasOwnProperty(word)) |
||||
wkw[word] = override[word]; |
||||
|
||||
return wkw; |
||||
})(); |
||||
|
||||
|
||||
|
||||
return { |
||||
startState: function () { return { f: normal }; }, |
||||
copyState: function (s) { return { f: s.f }; }, |
||||
|
||||
token: function(stream, state) { |
||||
var t = state.f(stream, function(s) { state.f = s; }); |
||||
var w = stream.current(); |
||||
return wellKnownWords.hasOwnProperty(w) ? wellKnownWords[w] : t; |
||||
}, |
||||
|
||||
blockCommentStart: "{-", |
||||
blockCommentEnd: "-}", |
||||
lineComment: "--" |
||||
}; |
||||
|
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-haskell", "haskell"); |
||||
|
||||
}); |
@ -0,0 +1,73 @@ |
||||
<!doctype html> |
||||
|
||||
<title>CodeMirror: Haskell mode</title> |
||||
<meta charset="utf-8"/> |
||||
<link rel=stylesheet href="../../doc/docs.css"> |
||||
|
||||
<link rel="stylesheet" href="../../lib/codemirror.css"> |
||||
<link rel="stylesheet" href="../../theme/elegant.css"> |
||||
<script src="../../lib/codemirror.js"></script> |
||||
<script src="../../addon/edit/matchbrackets.js"></script> |
||||
<script src="haskell.js"></script> |
||||
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style> |
||||
<div id=nav> |
||||
<a href="http://codemirror.net"><h1>CodeMirror</h1><img id=logo src="../../doc/logo.png"></a> |
||||
|
||||
<ul> |
||||
<li><a href="../../index.html">Home</a> |
||||
<li><a href="../../doc/manual.html">Manual</a> |
||||
<li><a href="https://github.com/codemirror/codemirror">Code</a> |
||||
</ul> |
||||
<ul> |
||||
<li><a href="../index.html">Language modes</a> |
||||
<li><a class=active href="#">Haskell</a> |
||||
</ul> |
||||
</div> |
||||
|
||||
<article> |
||||
<h2>Haskell mode</h2> |
||||
<form><textarea id="code" name="code"> |
||||
module UniquePerms ( |
||||
uniquePerms |
||||
) |
||||
where |
||||
|
||||
-- | Find all unique permutations of a list where there might be duplicates. |
||||
uniquePerms :: (Eq a) => [a] -> [[a]] |
||||
uniquePerms = permBag . makeBag |
||||
|
||||
-- | An unordered collection where duplicate values are allowed, |
||||
-- but represented with a single value and a count. |
||||
type Bag a = [(a, Int)] |
||||
|
||||
makeBag :: (Eq a) => [a] -> Bag a |
||||
makeBag [] = [] |
||||
makeBag (a:as) = mix a $ makeBag as |
||||
where |
||||
mix a [] = [(a,1)] |
||||
mix a (bn@(b,n):bs) | a == b = (b,n+1):bs |
||||
| otherwise = bn : mix a bs |
||||
|
||||
permBag :: Bag a -> [[a]] |
||||
permBag [] = [[]] |
||||
permBag bs = concatMap (\(f,cs) -> map (f:) $ permBag cs) . oneOfEach $ bs |
||||
where |
||||
oneOfEach [] = [] |
||||
oneOfEach (an@(a,n):bs) = |
||||
let bs' = if n == 1 then bs else (a,n-1):bs |
||||
in (a,bs') : mapSnd (an:) (oneOfEach bs) |
||||
|
||||
apSnd f (a,b) = (a, f b) |
||||
mapSnd = map . apSnd |
||||
</textarea></form> |
||||
|
||||
<script> |
||||
var editor = CodeMirror.fromTextArea(document.getElementById("code"), { |
||||
lineNumbers: true, |
||||
matchBrackets: true, |
||||
theme: "elegant" |
||||
}); |
||||
</script> |
||||
|
||||
<p><strong>MIME types defined:</strong> <code>text/x-haskell</code>.</p> |
||||
</article> |
@ -0,0 +1,515 @@ |
||||
// 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("haxe", function(config, parserConfig) { |
||||
var indentUnit = config.indentUnit; |
||||
|
||||
// Tokenizer
|
||||
|
||||
function kw(type) {return {type: type, style: "keyword"};} |
||||
var A = kw("keyword a"), B = kw("keyword b"), C = kw("keyword c"); |
||||
var operator = kw("operator"), atom = {type: "atom", style: "atom"}, attribute = {type:"attribute", style: "attribute"}; |
||||
var type = kw("typedef"); |
||||
var keywords = { |
||||
"if": A, "while": A, "else": B, "do": B, "try": B, |
||||
"return": C, "break": C, "continue": C, "new": C, "throw": C, |
||||
"var": kw("var"), "inline":attribute, "static": attribute, "using":kw("import"), |
||||
"public": attribute, "private": attribute, "cast": kw("cast"), "import": kw("import"), "macro": kw("macro"), |
||||
"function": kw("function"), "catch": kw("catch"), "untyped": kw("untyped"), "callback": kw("cb"), |
||||
"for": kw("for"), "switch": kw("switch"), "case": kw("case"), "default": kw("default"), |
||||
"in": operator, "never": kw("property_access"), "trace":kw("trace"), |
||||
"class": type, "abstract":type, "enum":type, "interface":type, "typedef":type, "extends":type, "implements":type, "dynamic":type, |
||||
"true": atom, "false": atom, "null": atom |
||||
}; |
||||
|
||||
var isOperatorChar = /[+\-*&%=<>!?|]/; |
||||
|
||||
function chain(stream, state, f) { |
||||
state.tokenize = f; |
||||
return f(stream, state); |
||||
} |
||||
|
||||
function toUnescaped(stream, end) { |
||||
var escaped = false, next; |
||||
while ((next = stream.next()) != null) { |
||||
if (next == end && !escaped) |
||||
return true; |
||||
escaped = !escaped && next == "\\"; |
||||
} |
||||
} |
||||
|
||||
// Used as scratch variables to communicate multiple values without
|
||||
// consing up tons of objects.
|
||||
var type, content; |
||||
function ret(tp, style, cont) { |
||||
type = tp; content = cont; |
||||
return style; |
||||
} |
||||
|
||||
function haxeTokenBase(stream, state) { |
||||
var ch = stream.next(); |
||||
if (ch == '"' || ch == "'") { |
||||
return chain(stream, state, haxeTokenString(ch)); |
||||
} else if (/[\[\]{}\(\),;\:\.]/.test(ch)) { |
||||
return ret(ch); |
||||
} else if (ch == "0" && stream.eat(/x/i)) { |
||||
stream.eatWhile(/[\da-f]/i); |
||||
return ret("number", "number"); |
||||
} else if (/\d/.test(ch) || ch == "-" && stream.eat(/\d/)) { |
||||
stream.match(/^\d*(?:\.\d*(?!\.))?(?:[eE][+\-]?\d+)?/); |
||||
return ret("number", "number"); |
||||
} else if (state.reAllowed && (ch == "~" && stream.eat(/\//))) { |
||||
toUnescaped(stream, "/"); |
||||
stream.eatWhile(/[gimsu]/); |
||||
return ret("regexp", "string-2"); |
||||
} else if (ch == "/") { |
||||
if (stream.eat("*")) { |
||||
return chain(stream, state, haxeTokenComment); |
||||
} else if (stream.eat("/")) { |
||||
stream.skipToEnd(); |
||||
return ret("comment", "comment"); |
||||
} else { |
||||
stream.eatWhile(isOperatorChar); |
||||
return ret("operator", null, stream.current()); |
||||
} |
||||
} else if (ch == "#") { |
||||
stream.skipToEnd(); |
||||
return ret("conditional", "meta"); |
||||
} else if (ch == "@") { |
||||
stream.eat(/:/); |
||||
stream.eatWhile(/[\w_]/); |
||||
return ret ("metadata", "meta"); |
||||
} else if (isOperatorChar.test(ch)) { |
||||
stream.eatWhile(isOperatorChar); |
||||
return ret("operator", null, stream.current()); |
||||
} else { |
||||
var word; |
||||
if(/[A-Z]/.test(ch)) { |
||||
stream.eatWhile(/[\w_<>]/); |
||||
word = stream.current(); |
||||
return ret("type", "variable-3", word); |
||||
} else { |
||||
stream.eatWhile(/[\w_]/); |
||||
var word = stream.current(), known = keywords.propertyIsEnumerable(word) && keywords[word]; |
||||
return (known && state.kwAllowed) ? ret(known.type, known.style, word) : |
||||
ret("variable", "variable", word); |
||||
} |
||||
} |
||||
} |
||||
|
||||
function haxeTokenString(quote) { |
||||
return function(stream, state) { |
||||
if (toUnescaped(stream, quote)) |
||||
state.tokenize = haxeTokenBase; |
||||
return ret("string", "string"); |
||||
}; |
||||
} |
||||
|
||||
function haxeTokenComment(stream, state) { |
||||
var maybeEnd = false, ch; |
||||
while (ch = stream.next()) { |
||||
if (ch == "/" && maybeEnd) { |
||||
state.tokenize = haxeTokenBase; |
||||
break; |
||||
} |
||||
maybeEnd = (ch == "*"); |
||||
} |
||||
return ret("comment", "comment"); |
||||
} |
||||
|
||||
// Parser
|
||||
|
||||
var atomicTypes = {"atom": true, "number": true, "variable": true, "string": true, "regexp": true}; |
||||
|
||||
function HaxeLexical(indented, column, type, align, prev, info) { |
||||
this.indented = indented; |
||||
this.column = column; |
||||
this.type = type; |
||||
this.prev = prev; |
||||
this.info = info; |
||||
if (align != null) this.align = align; |
||||
} |
||||
|
||||
function inScope(state, varname) { |
||||
for (var v = state.localVars; v; v = v.next) |
||||
if (v.name == varname) return true; |
||||
} |
||||
|
||||
function parseHaxe(state, style, type, content, stream) { |
||||
var cc = state.cc; |
||||
// Communicate our context to the combinators.
|
||||
// (Less wasteful than consing up a hundred closures on every call.)
|
||||
cx.state = state; cx.stream = stream; cx.marked = null, cx.cc = cc; |
||||
|
||||
if (!state.lexical.hasOwnProperty("align")) |
||||
state.lexical.align = true; |
||||
|
||||
while(true) { |
||||
var combinator = cc.length ? cc.pop() : statement; |
||||
if (combinator(type, content)) { |
||||
while(cc.length && cc[cc.length - 1].lex) |
||||
cc.pop()(); |
||||
if (cx.marked) return cx.marked; |
||||
if (type == "variable" && inScope(state, content)) return "variable-2"; |
||||
if (type == "variable" && imported(state, content)) return "variable-3"; |
||||
return style; |
||||
} |
||||
} |
||||
} |
||||
|
||||
function imported(state, typename) { |
||||
if (/[a-z]/.test(typename.charAt(0))) |
||||
return false; |
||||
var len = state.importedtypes.length; |
||||
for (var i = 0; i<len; i++) |
||||
if(state.importedtypes[i]==typename) return true; |
||||
} |
||||
|
||||
function registerimport(importname) { |
||||
var state = cx.state; |
||||
for (var t = state.importedtypes; t; t = t.next) |
||||
if(t.name == importname) return; |
||||
state.importedtypes = { name: importname, next: state.importedtypes }; |
||||
} |
||||
// Combinator utils
|
||||
|
||||
var cx = {state: null, column: null, marked: null, cc: null}; |
||||
function pass() { |
||||
for (var i = arguments.length - 1; i >= 0; i--) cx.cc.push(arguments[i]); |
||||
} |
||||
function cont() { |
||||
pass.apply(null, arguments); |
||||
return true; |
||||
} |
||||
function inList(name, list) { |
||||
for (var v = list; v; v = v.next) |
||||
if (v.name == name) return true; |
||||
return false; |
||||
} |
||||
function register(varname) { |
||||
var state = cx.state; |
||||
if (state.context) { |
||||
cx.marked = "def"; |
||||
if (inList(varname, state.localVars)) return; |
||||
state.localVars = {name: varname, next: state.localVars}; |
||||
} else if (state.globalVars) { |
||||
if (inList(varname, state.globalVars)) return; |
||||
state.globalVars = {name: varname, next: state.globalVars}; |
||||
} |
||||
} |
||||
|
||||
// Combinators
|
||||
|
||||
var defaultVars = {name: "this", next: null}; |
||||
function pushcontext() { |
||||
if (!cx.state.context) cx.state.localVars = defaultVars; |
||||
cx.state.context = {prev: cx.state.context, vars: cx.state.localVars}; |
||||
} |
||||
function popcontext() { |
||||
cx.state.localVars = cx.state.context.vars; |
||||
cx.state.context = cx.state.context.prev; |
||||
} |
||||
popcontext.lex = true; |
||||
function pushlex(type, info) { |
||||
var result = function() { |
||||
var state = cx.state; |
||||
state.lexical = new HaxeLexical(state.indented, cx.stream.column(), type, null, state.lexical, info); |
||||
}; |
||||
result.lex = true; |
||||
return result; |
||||
} |
||||
function poplex() { |
||||
var state = cx.state; |
||||
if (state.lexical.prev) { |
||||
if (state.lexical.type == ")") |
||||
state.indented = state.lexical.indented; |
||||
state.lexical = state.lexical.prev; |
||||
} |
||||
} |
||||
poplex.lex = true; |
||||
|
||||
function expect(wanted) { |
||||
function f(type) { |
||||
if (type == wanted) return cont(); |
||||
else if (wanted == ";") return pass(); |
||||
else return cont(f); |
||||
} |
||||
return f; |
||||
} |
||||
|
||||
function statement(type) { |
||||
if (type == "@") return cont(metadef); |
||||
if (type == "var") return cont(pushlex("vardef"), vardef1, expect(";"), poplex); |
||||
if (type == "keyword a") return cont(pushlex("form"), expression, statement, poplex); |
||||
if (type == "keyword b") return cont(pushlex("form"), statement, poplex); |
||||
if (type == "{") return cont(pushlex("}"), pushcontext, block, poplex, popcontext); |
||||
if (type == ";") return cont(); |
||||
if (type == "attribute") return cont(maybeattribute); |
||||
if (type == "function") return cont(functiondef); |
||||
if (type == "for") return cont(pushlex("form"), expect("("), pushlex(")"), forspec1, expect(")"), |
||||
poplex, statement, poplex); |
||||
if (type == "variable") return cont(pushlex("stat"), maybelabel); |
||||
if (type == "switch") return cont(pushlex("form"), expression, pushlex("}", "switch"), expect("{"), |
||||
block, poplex, poplex); |
||||
if (type == "case") return cont(expression, expect(":")); |
||||
if (type == "default") return cont(expect(":")); |
||||
if (type == "catch") return cont(pushlex("form"), pushcontext, expect("("), funarg, expect(")"), |
||||
statement, poplex, popcontext); |
||||
if (type == "import") return cont(importdef, expect(";")); |
||||
if (type == "typedef") return cont(typedef); |
||||
return pass(pushlex("stat"), expression, expect(";"), poplex); |
||||
} |
||||
function expression(type) { |
||||
if (atomicTypes.hasOwnProperty(type)) return cont(maybeoperator); |
||||
if (type == "type" ) return cont(maybeoperator); |
||||
if (type == "function") return cont(functiondef); |
||||
if (type == "keyword c") return cont(maybeexpression); |
||||
if (type == "(") return cont(pushlex(")"), maybeexpression, expect(")"), poplex, maybeoperator); |
||||
if (type == "operator") return cont(expression); |
||||
if (type == "[") return cont(pushlex("]"), commasep(maybeexpression, "]"), poplex, maybeoperator); |
||||
if (type == "{") return cont(pushlex("}"), commasep(objprop, "}"), poplex, maybeoperator); |
||||
return cont(); |
||||
} |
||||
function maybeexpression(type) { |
||||
if (type.match(/[;\}\)\],]/)) return pass(); |
||||
return pass(expression); |
||||
} |
||||
|
||||
function maybeoperator(type, value) { |
||||
if (type == "operator" && /\+\+|--/.test(value)) return cont(maybeoperator); |
||||
if (type == "operator" || type == ":") return cont(expression); |
||||
if (type == ";") return; |
||||
if (type == "(") return cont(pushlex(")"), commasep(expression, ")"), poplex, maybeoperator); |
||||
if (type == ".") return cont(property, maybeoperator); |
||||
if (type == "[") return cont(pushlex("]"), expression, expect("]"), poplex, maybeoperator); |
||||
} |
||||
|
||||
function maybeattribute(type) { |
||||
if (type == "attribute") return cont(maybeattribute); |
||||
if (type == "function") return cont(functiondef); |
||||
if (type == "var") return cont(vardef1); |
||||
} |
||||
|
||||
function metadef(type) { |
||||
if(type == ":") return cont(metadef); |
||||
if(type == "variable") return cont(metadef); |
||||
if(type == "(") return cont(pushlex(")"), commasep(metaargs, ")"), poplex, statement); |
||||
} |
||||
function metaargs(type) { |
||||
if(type == "variable") return cont(); |
||||
} |
||||
|
||||
function importdef (type, value) { |
||||
if(type == "variable" && /[A-Z]/.test(value.charAt(0))) { registerimport(value); return cont(); } |
||||
else if(type == "variable" || type == "property" || type == "." || value == "*") return cont(importdef); |
||||
} |
||||
|
||||
function typedef (type, value) |
||||
{ |
||||
if(type == "variable" && /[A-Z]/.test(value.charAt(0))) { registerimport(value); return cont(); } |
||||
else if (type == "type" && /[A-Z]/.test(value.charAt(0))) { return cont(); } |
||||
} |
||||
|
||||
function maybelabel(type) { |
||||
if (type == ":") return cont(poplex, statement); |
||||
return pass(maybeoperator, expect(";"), poplex); |
||||
} |
||||
function property(type) { |
||||
if (type == "variable") {cx.marked = "property"; return cont();} |
||||
} |
||||
function objprop(type) { |
||||
if (type == "variable") cx.marked = "property"; |
||||
if (atomicTypes.hasOwnProperty(type)) return cont(expect(":"), expression); |
||||
} |
||||
function commasep(what, end) { |
||||
function proceed(type) { |
||||
if (type == ",") return cont(what, proceed); |
||||
if (type == end) return cont(); |
||||
return cont(expect(end)); |
||||
} |
||||
return function(type) { |
||||
if (type == end) return cont(); |
||||
else return pass(what, proceed); |
||||
}; |
||||
} |
||||
function block(type) { |
||||
if (type == "}") return cont(); |
||||
return pass(statement, block); |
||||
} |
||||
function vardef1(type, value) { |
||||
if (type == "variable"){register(value); return cont(typeuse, vardef2);} |
||||
return cont(); |
||||
} |
||||
function vardef2(type, value) { |
||||
if (value == "=") return cont(expression, vardef2); |
||||
if (type == ",") return cont(vardef1); |
||||
} |
||||
function forspec1(type, value) { |
||||
if (type == "variable") { |
||||
register(value); |
||||
return cont(forin, expression) |
||||
} else { |
||||
return pass() |
||||
} |
||||
} |
||||
function forin(_type, value) { |
||||
if (value == "in") return cont(); |
||||
} |
||||
function functiondef(type, value) { |
||||
//function names starting with upper-case letters are recognised as types, so cludging them together here.
|
||||
if (type == "variable" || type == "type") {register(value); return cont(functiondef);} |
||||
if (value == "new") return cont(functiondef); |
||||
if (type == "(") return cont(pushlex(")"), pushcontext, commasep(funarg, ")"), poplex, typeuse, statement, popcontext); |
||||
} |
||||
function typeuse(type) { |
||||
if(type == ":") return cont(typestring); |
||||
} |
||||
function typestring(type) { |
||||
if(type == "type") return cont(); |
||||
if(type == "variable") return cont(); |
||||
if(type == "{") return cont(pushlex("}"), commasep(typeprop, "}"), poplex); |
||||
} |
||||
function typeprop(type) { |
||||
if(type == "variable") return cont(typeuse); |
||||
} |
||||
function funarg(type, value) { |
||||
if (type == "variable") {register(value); return cont(typeuse);} |
||||
} |
||||
|
||||
// Interface
|
||||
return { |
||||
startState: function(basecolumn) { |
||||
var defaulttypes = ["Int", "Float", "String", "Void", "Std", "Bool", "Dynamic", "Array"]; |
||||
var state = { |
||||
tokenize: haxeTokenBase, |
||||
reAllowed: true, |
||||
kwAllowed: true, |
||||
cc: [], |
||||
lexical: new HaxeLexical((basecolumn || 0) - indentUnit, 0, "block", false), |
||||
localVars: parserConfig.localVars, |
||||
importedtypes: defaulttypes, |
||||
context: parserConfig.localVars && {vars: parserConfig.localVars}, |
||||
indented: 0 |
||||
}; |
||||
if (parserConfig.globalVars && typeof parserConfig.globalVars == "object") |
||||
state.globalVars = parserConfig.globalVars; |
||||
return state; |
||||
}, |
||||
|
||||
token: function(stream, state) { |
||||
if (stream.sol()) { |
||||
if (!state.lexical.hasOwnProperty("align")) |
||||
state.lexical.align = false; |
||||
state.indented = stream.indentation(); |
||||
} |
||||
if (stream.eatSpace()) return null; |
||||
var style = state.tokenize(stream, state); |
||||
if (type == "comment") return style; |
||||
state.reAllowed = !!(type == "operator" || type == "keyword c" || type.match(/^[\[{}\(,;:]$/)); |
||||
state.kwAllowed = type != '.'; |
||||
return parseHaxe(state, style, type, content, stream); |
||||
}, |
||||
|
||||
indent: function(state, textAfter) { |
||||
if (state.tokenize != haxeTokenBase) return 0; |
||||
var firstChar = textAfter && textAfter.charAt(0), lexical = state.lexical; |
||||
if (lexical.type == "stat" && firstChar == "}") lexical = lexical.prev; |
||||
var type = lexical.type, closing = firstChar == type; |
||||
if (type == "vardef") return lexical.indented + 4; |
||||
else if (type == "form" && firstChar == "{") return lexical.indented; |
||||
else if (type == "stat" || type == "form") return lexical.indented + indentUnit; |
||||
else if (lexical.info == "switch" && !closing) |
||||
return lexical.indented + (/^(?:case|default)\b/.test(textAfter) ? indentUnit : 2 * indentUnit); |
||||
else if (lexical.align) return lexical.column + (closing ? 0 : 1); |
||||
else return lexical.indented + (closing ? 0 : indentUnit); |
||||
}, |
||||
|
||||
electricChars: "{}", |
||||
blockCommentStart: "/*", |
||||
blockCommentEnd: "*/", |
||||
lineComment: "//" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-haxe", "haxe"); |
||||
|
||||
CodeMirror.defineMode("hxml", function () { |
||||
|
||||
return { |
||||
startState: function () { |
||||
return { |
||||
define: false, |
||||
inString: false |
||||
}; |
||||
}, |
||||
token: function (stream, state) { |
||||
var ch = stream.peek(); |
||||
var sol = stream.sol(); |
||||
|
||||
///* comments */
|
||||
if (ch == "#") { |
||||
stream.skipToEnd(); |
||||
return "comment"; |
||||
} |
||||
if (sol && ch == "-") { |
||||
var style = "variable-2"; |
||||
|
||||
stream.eat(/-/); |
||||
|
||||
if (stream.peek() == "-") { |
||||
stream.eat(/-/); |
||||
style = "keyword a"; |
||||
} |
||||
|
||||
if (stream.peek() == "D") { |
||||
stream.eat(/[D]/); |
||||
style = "keyword c"; |
||||
state.define = true; |
||||
} |
||||
|
||||
stream.eatWhile(/[A-Z]/i); |
||||
return style; |
||||
} |
||||
|
||||
var ch = stream.peek(); |
||||
|
||||
if (state.inString == false && ch == "'") { |
||||
state.inString = true; |
||||
ch = stream.next(); |
||||
} |
||||
|
||||
if (state.inString == true) { |
||||
if (stream.skipTo("'")) { |
||||
|
||||
} else { |
||||
stream.skipToEnd(); |
||||
} |
||||
|
||||
if (stream.peek() == "'") { |
||||
stream.next(); |
||||
state.inString = false; |
||||
} |
||||
|
||||
return "string"; |
||||
} |
||||
|
||||
stream.next(); |
||||
return null; |
||||
}, |
||||
lineComment: "#" |
||||
}; |
||||
}); |
||||
|
||||
CodeMirror.defineMIME("text/x-hxml", "hxml"); |
||||
|
||||
}); |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue