Replace textarea instaces CodeMirror
Added CodeMirror files TODO: remove unnecessary files
This commit is contained in:
parent
5623fef058
commit
b87ab87ce0
2
codemirror-2.36/.gitignore
vendored
Normal file
2
codemirror-2.36/.gitignore
vendored
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
/node_modules
|
||||||
|
/npm-debug.log
|
3
codemirror-2.36/.travis.yml
Normal file
3
codemirror-2.36/.travis.yml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
language: node_js
|
||||||
|
node_js:
|
||||||
|
- 0.8
|
23
codemirror-2.36/LICENSE
Normal file
23
codemirror-2.36/LICENSE
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
Copyright (C) 2012 by Marijn Haverbeke <marijnh@gmail.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
|
||||||
|
Please note that some subdirectories of the CodeMirror distribution
|
||||||
|
include their own LICENSE files, and are released under different
|
||||||
|
licences.
|
8
codemirror-2.36/README.md
Normal file
8
codemirror-2.36/README.md
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
# CodeMirror [![Build Status](https://secure.travis-ci.org/marijnh/CodeMirror.png?branch=master)](http://travis-ci.org/marijnh/CodeMirror)
|
||||||
|
|
||||||
|
CodeMirror is a JavaScript component that provides a code editor in
|
||||||
|
the browser. When a mode is available for the language you are coding
|
||||||
|
in, it will color your code, and optionally help with indentation.
|
||||||
|
|
||||||
|
The project page is http://codemirror.net
|
||||||
|
The manual is at http://codemirror.net/doc/manual.html
|
88
codemirror-2.36/bin/compress
Executable file
88
codemirror-2.36/bin/compress
Executable file
|
@ -0,0 +1,88 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
// Compression helper for CodeMirror
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// bin/compress codemirror runmode javascript xml
|
||||||
|
//
|
||||||
|
// Will take lib/codemirror.js, lib/util/runmode.js,
|
||||||
|
// mode/javascript/javascript.js, and mode/xml/xml.js, run them though
|
||||||
|
// the online minifier at http://marijnhaverbeke.nl/uglifyjs, and spit
|
||||||
|
// out the result.
|
||||||
|
//
|
||||||
|
// bin/compress codemirror --local /path/to/bin/UglifyJS
|
||||||
|
//
|
||||||
|
// Will use a local minifier instead of the online default one.
|
||||||
|
//
|
||||||
|
// Script files are specified without .js ending. Prefixing them with
|
||||||
|
// their full (local) path is optional. So you may say lib/codemirror
|
||||||
|
// or mode/xml/xml to be more precise. In fact, even the .js suffix
|
||||||
|
// may be speficied, if wanted.
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var fs = require("fs");
|
||||||
|
|
||||||
|
function help(ok) {
|
||||||
|
console.log("usage: " + process.argv[1] + " [--local /path/to/uglifyjs] files...");
|
||||||
|
process.exit(ok ? 0 : 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
var local = null, args = null, files = [], blob = "";
|
||||||
|
|
||||||
|
for (var i = 2; i < process.argv.length; ++i) {
|
||||||
|
var arg = process.argv[i];
|
||||||
|
if (arg == "--local" && i + 1 < process.argv.length) {
|
||||||
|
var parts = process.argv[++i].split(/\s+/);
|
||||||
|
local = parts[0];
|
||||||
|
args = parts.slice(1);
|
||||||
|
} else if (arg == "--help") {
|
||||||
|
help(true);
|
||||||
|
} else if (arg[0] != "-") {
|
||||||
|
files.push({name: arg, re: new RegExp("(?:\\/|^)" + arg + (/\.js$/.test(arg) ? "$" : "\\.js$"))});
|
||||||
|
} else help(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
function walk(dir) {
|
||||||
|
fs.readdirSync(dir).forEach(function(fname) {
|
||||||
|
if (/^[_\.]/.test(fname)) return;
|
||||||
|
var file = dir + fname;
|
||||||
|
if (fs.statSync(file).isDirectory()) return walk(file + "/");
|
||||||
|
if (files.some(function(spec, i) {
|
||||||
|
var match = spec.re.test(file);
|
||||||
|
if (match) files.splice(i, 1);
|
||||||
|
return match;
|
||||||
|
})) {
|
||||||
|
if (local) args.push(file);
|
||||||
|
else blob += fs.readFileSync(file, "utf8");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
walk("lib/");
|
||||||
|
walk("mode/");
|
||||||
|
|
||||||
|
if (files.length) {
|
||||||
|
console.log("Some speficied files were not found: " +
|
||||||
|
files.map(function(a){return a.name;}).join(", "));
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (local) {
|
||||||
|
require("child_process").spawn(local, args, {stdio: ["ignore", process.stdout, process.stderr]});
|
||||||
|
} else {
|
||||||
|
var data = new Buffer("js_code=" + require("querystring").escape(blob), "utf8");
|
||||||
|
var req = require("http").request({
|
||||||
|
host: "marijnhaverbeke.nl",
|
||||||
|
port: 80,
|
||||||
|
method: "POST",
|
||||||
|
path: "/uglifyjs",
|
||||||
|
headers: {"content-type": "application/x-www-form-urlencoded",
|
||||||
|
"content-length": data.length}
|
||||||
|
});
|
||||||
|
req.on("response", function(resp) {
|
||||||
|
resp.on("data", function (chunk) { process.stdout.write(chunk); });
|
||||||
|
});
|
||||||
|
req.end(data);
|
||||||
|
}
|
73
codemirror-2.36/demo/activeline.html
Normal file
73
codemirror-2.36/demo/activeline.html
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Active Line Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}
|
||||||
|
.activeline {background: #e8f2ff !important;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Active Line Demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0"
|
||||||
|
xmlns:georss="http://www.georss.org/georss"
|
||||||
|
xmlns:twitter="http://api.twitter.com">
|
||||||
|
<channel>
|
||||||
|
<title>Twitter / codemirror</title>
|
||||||
|
<link>http://twitter.com/codemirror</link>
|
||||||
|
<atom:link type="application/rss+xml"
|
||||||
|
href="http://twitter.com/statuses/user_timeline/242283288.rss" rel="self"/>
|
||||||
|
<description>Twitter updates from CodeMirror / codemirror.</description>
|
||||||
|
<language>en-us</language>
|
||||||
|
<ttl>40</ttl>
|
||||||
|
<item>
|
||||||
|
<title>codemirror: http://cloud-ide.com — they're springing up like mushrooms. This one
|
||||||
|
uses CodeMirror as its editor.</title>
|
||||||
|
<description>codemirror: http://cloud-ide.com — they're springing up like mushrooms. This
|
||||||
|
one uses CodeMirror as its editor.</description>
|
||||||
|
<pubDate>Thu, 17 Mar 2011 23:34:47 +0000</pubDate>
|
||||||
|
<guid>http://twitter.com/codemirror/statuses/48527733722058752</guid>
|
||||||
|
<link>http://twitter.com/codemirror/statuses/48527733722058752</link>
|
||||||
|
<twitter:source>web</twitter:source>
|
||||||
|
<twitter:place/>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<title>codemirror: Posted a description of the CodeMirror 2 internals at
|
||||||
|
http://codemirror.net/2/internals.html</title>
|
||||||
|
<description>codemirror: Posted a description of the CodeMirror 2 internals at
|
||||||
|
http://codemirror.net/2/internals.html</description>
|
||||||
|
<pubDate>Wed, 02 Mar 2011 12:15:09 +0000</pubDate>
|
||||||
|
<guid>http://twitter.com/codemirror/statuses/42920879788789760</guid>
|
||||||
|
<link>http://twitter.com/codemirror/statuses/42920879788789760</link>
|
||||||
|
<twitter:source>web</twitter:source>
|
||||||
|
<twitter:place/>
|
||||||
|
</item>
|
||||||
|
</channel>
|
||||||
|
</rss></textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
mode: "application/xml",
|
||||||
|
lineNumbers: true,
|
||||||
|
lineWrapping: true,
|
||||||
|
onCursorActivity: function() {
|
||||||
|
editor.setLineClass(hlLine, null, null);
|
||||||
|
hlLine = editor.setLineClass(editor.getCursor().line, null, "activeline");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
var hlLine = editor.setLineClass(0, "activeline");
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Styling the current cursor line.</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
51
codemirror-2.36/demo/changemode.html
Normal file
51
codemirror-2.36/demo/changemode.html
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Mode-Changing Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<script src="../mode/scheme/scheme.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border: 1px solid black;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Mode-Changing demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
;; If there is Scheme code in here, the editor will be in Scheme mode.
|
||||||
|
;; If you put in JS instead, it'll switch to JS mode.
|
||||||
|
|
||||||
|
(define (double x)
|
||||||
|
(* x x))
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>On changes to the content of the above editor, a (crude) script
|
||||||
|
tries to auto-detect the language used, and switches the editor to
|
||||||
|
either JavaScript or Scheme mode based on that.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
mode: "scheme",
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
tabMode: "indent",
|
||||||
|
onChange: function() {
|
||||||
|
clearTimeout(pending);
|
||||||
|
setTimeout(update, 400);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
var pending;
|
||||||
|
function looksLikeScheme(code) {
|
||||||
|
return !/^\s*\(\s*function\b/.test(code) && /^\s*[;\(]/.test(code);
|
||||||
|
}
|
||||||
|
function update() {
|
||||||
|
editor.setOption("mode", looksLikeScheme(editor.getValue()) ? "scheme" : "javascript");
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
66
codemirror-2.36/demo/closetag.html
Normal file
66
codemirror-2.36/demo/closetag.html
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Close-Tag Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/closetag.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<script src="../mode/css/css.js"></script>
|
||||||
|
<script src="../mode/htmlmixed/htmlmixed.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid #eee; border-bottom: 1px solid #eee;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1>Close-Tag Demo</h1>
|
||||||
|
<ul>
|
||||||
|
<li>Type an html tag. When you type '>' or '/', the tag will auto-close/complete. Block-level tags will indent.</li>
|
||||||
|
<li>There are options for disabling tag closing or customizing the list of tags to indent.</li>
|
||||||
|
<li>Works with "text/html" (based on htmlmixed.js or xml.js) mode.</li>
|
||||||
|
<li>View source for key binding details.</li>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code"></textarea></form>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
mode: 'text/html',
|
||||||
|
|
||||||
|
//closeTagEnabled: false, // Set this option to disable tag closing behavior without having to remove the key bindings.
|
||||||
|
//closeTagIndent: false, // Pass false or an array of tag names to override the default indentation behavior.
|
||||||
|
|
||||||
|
extraKeys: {
|
||||||
|
"'>'": function(cm) { cm.closeTag(cm, '>'); },
|
||||||
|
"'/'": function(cm) { cm.closeTag(cm, '/'); }
|
||||||
|
},
|
||||||
|
|
||||||
|
/*
|
||||||
|
// extraKeys is the easier way to go, but if you need native key event processing, this should work too.
|
||||||
|
onKeyEvent: function(cm, e) {
|
||||||
|
if (e.type == 'keydown') {
|
||||||
|
var c = e.keyCode || e.which;
|
||||||
|
if (c == 190 || c == 191) {
|
||||||
|
try {
|
||||||
|
cm.closeTag(cm, c == 190 ? '>' : '/');
|
||||||
|
e.stop();
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
if (e != CodeMirror.Pass) throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
*/
|
||||||
|
|
||||||
|
wordWrap: true
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
71
codemirror-2.36/demo/complete.html
Normal file
71
codemirror-2.36/demo/complete.html
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Autocomplete Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/simple-hint.js"></script>
|
||||||
|
<link rel="stylesheet" href="../lib/util/simple-hint.css">
|
||||||
|
<script src="../lib/util/javascript-hint.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
<style type="text/css">.CodeMirror {border: 1px solid #eee;} .CodeMirror-scroll { height: 100% }</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Autocomplete demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
function getCompletions(token, context) {
|
||||||
|
var found = [], start = token.string;
|
||||||
|
function maybeAdd(str) {
|
||||||
|
if (str.indexOf(start) == 0) found.push(str);
|
||||||
|
}
|
||||||
|
function gatherCompletions(obj) {
|
||||||
|
if (typeof obj == "string") forEach(stringProps, maybeAdd);
|
||||||
|
else if (obj instanceof Array) forEach(arrayProps, maybeAdd);
|
||||||
|
else if (obj instanceof Function) forEach(funcProps, maybeAdd);
|
||||||
|
for (var name in obj) maybeAdd(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (context) {
|
||||||
|
// If this is a property, see if it belongs to some object we can
|
||||||
|
// find in the current environment.
|
||||||
|
var obj = context.pop(), base;
|
||||||
|
if (obj.className == "js-variable")
|
||||||
|
base = window[obj.string];
|
||||||
|
else if (obj.className == "js-string")
|
||||||
|
base = "";
|
||||||
|
else if (obj.className == "js-atom")
|
||||||
|
base = 1;
|
||||||
|
while (base != null && context.length)
|
||||||
|
base = base[context.pop().string];
|
||||||
|
if (base != null) gatherCompletions(base);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// If not, just look in the window object and any local scope
|
||||||
|
// (reading into JS mode internals to get at the local variables)
|
||||||
|
for (var v = token.state.localVars; v; v = v.next) maybeAdd(v.name);
|
||||||
|
gatherCompletions(window);
|
||||||
|
forEach(keywords, maybeAdd);
|
||||||
|
}
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>Press <strong>ctrl-space</strong> to activate autocompletion. See
|
||||||
|
the code (<a href="../lib/util/simple-hint.js">here</a>
|
||||||
|
and <a href="../lib/util/javascript-hint.js">here</a>) to figure out
|
||||||
|
how it works.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.commands.autocomplete = function(cm) {
|
||||||
|
CodeMirror.simpleHint(cm, CodeMirror.javascriptHint);
|
||||||
|
}
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
extraKeys: {"Ctrl-Space": "autocomplete"}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
60
codemirror-2.36/demo/emacs.html
Normal file
60
codemirror-2.36/demo/emacs.html
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Emacs bindings demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/clike/clike.js"></script>
|
||||||
|
<script src="../keymap/emacs.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid #eee; border-bottom: 1px solid #eee;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Emacs bindings demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
#include "syscalls.h"
|
||||||
|
/* getchar: simple buffered version */
|
||||||
|
int getchar(void)
|
||||||
|
{
|
||||||
|
static char buf[BUFSIZ];
|
||||||
|
static char *bufp = buf;
|
||||||
|
static int n = 0;
|
||||||
|
if (n == 0) { /* buffer is empty */
|
||||||
|
n = read(0, buf, sizeof buf);
|
||||||
|
bufp = buf;
|
||||||
|
}
|
||||||
|
return (--n >= 0) ? (unsigned char) *bufp++ : EOF;
|
||||||
|
}
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>The emacs keybindings are enabled by
|
||||||
|
including <a href="../keymap/emacs.js">keymap/emacs.js</a> and setting
|
||||||
|
the <code>keyMap</code> option to <code>"emacs"</code>. Because
|
||||||
|
CodeMirror's internal API is quite different from Emacs, they are only
|
||||||
|
a loose approximation of actual emacs bindings, though.</p>
|
||||||
|
|
||||||
|
<p>Also note that a lot of browsers disallow certain keys from being
|
||||||
|
captured. For example, Chrome blocks both Ctrl-W and Ctrl-N, with the
|
||||||
|
result that idiomatic use of Emacs keys will constantly close your tab
|
||||||
|
or open a new window.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.commands.save = function() {
|
||||||
|
var elt = editor.getWrapperElement();
|
||||||
|
elt.style.background = "#def";
|
||||||
|
setTimeout(function() { elt.style.background = ""; }, 300);
|
||||||
|
};
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
mode: "text/x-csrc",
|
||||||
|
keyMap: "emacs"
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
62
codemirror-2.36/demo/folding.html
Normal file
62
codemirror-2.36/demo/folding.html
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Code Folding Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/foldcode.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}
|
||||||
|
.CodeMirror-gutter {min-width: 2.6em; cursor: pointer;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Code Folding Demo</h1>
|
||||||
|
|
||||||
|
<p>Demonstration of code folding using the code
|
||||||
|
in <a href="../lib/util/foldcode.js"><code>foldcode.js</code></a>.
|
||||||
|
Press ctrl-q or click on the gutter to fold a block, again
|
||||||
|
to unfold.</p>
|
||||||
|
<form>
|
||||||
|
<div style="max-width: 50em; margin-bottom: 1em">JavaScript:<br><textarea id="code" name="code"></textarea></div>
|
||||||
|
<div style="max-width: 50em">HTML:<br><textarea id="code-html" name="code-html"></textarea></div>
|
||||||
|
</form>
|
||||||
|
<script id="script">
|
||||||
|
window.onload = function() {
|
||||||
|
var te = document.getElementById("code");
|
||||||
|
var sc = document.getElementById("script");
|
||||||
|
te.value = (sc.textContent || sc.innerText || sc.innerHTML).replace(/^\s*/, "");
|
||||||
|
sc.innerHTML = "";
|
||||||
|
var te_html = document.getElementById("code-html");
|
||||||
|
te_html.value = "<html>\n " + document.documentElement.innerHTML + "\n</html>";
|
||||||
|
|
||||||
|
var foldFunc = CodeMirror.newFoldFunction(CodeMirror.braceRangeFinder);
|
||||||
|
window.editor = CodeMirror.fromTextArea(te, {
|
||||||
|
mode: "javascript",
|
||||||
|
lineNumbers: true,
|
||||||
|
lineWrapping: true,
|
||||||
|
onGutterClick: foldFunc,
|
||||||
|
extraKeys: {"Ctrl-Q": function(cm){foldFunc(cm, cm.getCursor().line);}}
|
||||||
|
});
|
||||||
|
foldFunc(editor, 9);
|
||||||
|
foldFunc(editor, 20);
|
||||||
|
|
||||||
|
var foldFunc_html = CodeMirror.newFoldFunction(CodeMirror.tagRangeFinder);
|
||||||
|
window.editor_html = CodeMirror.fromTextArea(te_html, {
|
||||||
|
mode: "text/html",
|
||||||
|
lineNumbers: true,
|
||||||
|
lineWrapping: true,
|
||||||
|
onGutterClick: foldFunc_html,
|
||||||
|
extraKeys: {"Ctrl-Q": function(cm){foldFunc_html(cm, cm.getCursor().line);}}
|
||||||
|
})
|
||||||
|
foldFunc_html(editor_html, 1);
|
||||||
|
foldFunc_html(editor_html, 15);
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
81
codemirror-2.36/demo/formatting.html
Normal file
81
codemirror-2.36/demo/formatting.html
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Formatting Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/formatting.js"></script>
|
||||||
|
<script src="../mode/css/css.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<script src="../mode/htmlmixed/htmlmixed.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {
|
||||||
|
border: 1px solid #eee;
|
||||||
|
}
|
||||||
|
td {
|
||||||
|
padding-right: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Formatting demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code"><script> function (s,e){ for(var i=0; i < 1; i++) test("test();a=1");} </script>
|
||||||
|
<script>
|
||||||
|
function test(c){ for (var i = 0; i < 10; i++){ process("a.b();c = null;", 300);}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<table><tr><td>test 1</td></tr><tr><td>test 2</td></tr></table>
|
||||||
|
<script> function test() { return 1;} </script>
|
||||||
|
<style> .test { font-size: medium; font-family: monospace; }
|
||||||
|
</style></textarea></form>
|
||||||
|
|
||||||
|
<p>Select a piece of code and click one of the links below to apply automatic formatting to the selected text or comment/uncomment the selected text. Note that the formatting behavior depends on the current block's mode.
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<a href="javascript:autoFormatSelection()">
|
||||||
|
Autoformat Selected
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<a href="javascript:commentSelection(true)">
|
||||||
|
Comment Selected
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<a href="javascript:commentSelection(false)">
|
||||||
|
Uncomment Selected
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</p>
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
mode: "htmlmixed"
|
||||||
|
});
|
||||||
|
CodeMirror.commands["selectAll"](editor);
|
||||||
|
|
||||||
|
function getSelectedRange() {
|
||||||
|
return { from: editor.getCursor(true), to: editor.getCursor(false) };
|
||||||
|
}
|
||||||
|
|
||||||
|
function autoFormatSelection() {
|
||||||
|
var range = getSelectedRange();
|
||||||
|
editor.autoFormatRange(range.from, range.to);
|
||||||
|
}
|
||||||
|
|
||||||
|
function commentSelection(isComment) {
|
||||||
|
var range = getSelectedRange();
|
||||||
|
editor.commentRange(isComment, range.from, range.to);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
147
codemirror-2.36/demo/fullscreen.html
Normal file
147
codemirror-2.36/demo/fullscreen.html
Normal file
|
@ -0,0 +1,147 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Full Screen Editing</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<link rel="stylesheet" href="../theme/night.css">
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror-fullscreen {
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
top: 0; left: 0;
|
||||||
|
width: 100%;
|
||||||
|
z-index: 9999;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Full Screen Editing</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code" rows="5">
|
||||||
|
<dt id="option_indentWithTabs"><code>indentWithTabs (boolean)</code></dt>
|
||||||
|
<dd>Whether, when indenting, the first N*8 spaces should be
|
||||||
|
replaced by N tabs. Default is false.</dd>
|
||||||
|
|
||||||
|
<dt id="option_tabMode"><code>tabMode (string)</code></dt>
|
||||||
|
<dd>Determines what happens when the user presses the tab key.
|
||||||
|
Must be one of the following:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"classic" (the default)</code></dt>
|
||||||
|
<dd>When nothing is selected, insert a tab. Otherwise,
|
||||||
|
behave like the <code>"shift"</code> mode. (When shift is
|
||||||
|
held, this behaves like the <code>"indent"</code> mode.)</dd>
|
||||||
|
<dt><code>"shift"</code></dt>
|
||||||
|
<dd>Indent all selected lines by
|
||||||
|
one <a href="#option_indentUnit"><code>indentUnit</code></a>.
|
||||||
|
If shift was held while pressing tab, un-indent all selected
|
||||||
|
lines one unit.</dd>
|
||||||
|
<dt><code>"indent"</code></dt>
|
||||||
|
<dd>Indent the line the 'correctly', based on its syntactic
|
||||||
|
context. Only works if the
|
||||||
|
mode <a href="#indent">supports</a> it.</dd>
|
||||||
|
<dt><code>"default"</code></dt>
|
||||||
|
<dd>Do not capture tab presses, let the browser apply its
|
||||||
|
default behaviour (which usually means it skips to the next
|
||||||
|
control).</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
<dt id="option_enterMode"><code>enterMode (string)</code></dt>
|
||||||
|
<dd>Determines whether and how new lines are indented when the
|
||||||
|
enter key is pressed. The following modes are supported:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"indent" (the default)</code></dt>
|
||||||
|
<dd>Use the mode's indentation rules to give the new line
|
||||||
|
the correct indentation.</dd>
|
||||||
|
<dt><code>"keep"</code></dt>
|
||||||
|
<dd>Indent the line the same as the previous line.</dd>
|
||||||
|
<dt><code>"flat"</code></dt>
|
||||||
|
<dd>Do not indent the new line.</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
<dt id="option_enterMode"><code>enterMode (string)</code></dt>
|
||||||
|
<dd>Determines whether and how new lines are indented when the
|
||||||
|
enter key is pressed. The following modes are supported:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"indent" (the default)</code></dt>
|
||||||
|
<dd>Use the mode's indentation rules to give the new line
|
||||||
|
the correct indentation.</dd>
|
||||||
|
<dt><code>"keep"</code></dt>
|
||||||
|
<dd>Indent the line the same as the previous line.</dd>
|
||||||
|
<dt><code>"flat"</code></dt>
|
||||||
|
<dd>Do not indent the new line.</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
<dt id="option_enterMode"><code>enterMode (string)</code></dt>
|
||||||
|
<dd>Determines whether and how new lines are indented when the
|
||||||
|
enter key is pressed. The following modes are supported:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"indent" (the default)</code></dt>
|
||||||
|
<dd>Use the mode's indentation rules to give the new line
|
||||||
|
the correct indentation.</dd>
|
||||||
|
<dt><code>"keep"</code></dt>
|
||||||
|
<dd>Indent the line the same as the previous line.</dd>
|
||||||
|
<dt><code>"flat"</code></dt>
|
||||||
|
<dd>Do not indent the new line.</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
<dt id="option_enterMode"><code>enterMode (string)</code></dt>
|
||||||
|
<dd>Determines whether and how new lines are indented when the
|
||||||
|
enter key is pressed. The following modes are supported:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"indent" (the default)</code></dt>
|
||||||
|
<dd>Use the mode's indentation rules to give the new line
|
||||||
|
the correct indentation.</dd>
|
||||||
|
<dt><code>"keep"</code></dt>
|
||||||
|
<dd>Indent the line the same as the previous line.</dd>
|
||||||
|
<dt><code>"flat"</code></dt>
|
||||||
|
<dd>Do not indent the new line.</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
</textarea></form>
|
||||||
|
<script>
|
||||||
|
function isFullScreen(cm) {
|
||||||
|
return /\bCodeMirror-fullscreen\b/.test(cm.getWrapperElement().className);
|
||||||
|
}
|
||||||
|
function winHeight() {
|
||||||
|
return window.innerHeight || (document.documentElement || document.body).clientHeight;
|
||||||
|
}
|
||||||
|
function setFullScreen(cm, full) {
|
||||||
|
var wrap = cm.getWrapperElement(), scroll = cm.getScrollerElement();
|
||||||
|
if (full) {
|
||||||
|
wrap.className += " CodeMirror-fullscreen";
|
||||||
|
scroll.style.height = winHeight() + "px";
|
||||||
|
document.documentElement.style.overflow = "hidden";
|
||||||
|
} else {
|
||||||
|
wrap.className = wrap.className.replace(" CodeMirror-fullscreen", "");
|
||||||
|
scroll.style.height = "";
|
||||||
|
document.documentElement.style.overflow = "";
|
||||||
|
}
|
||||||
|
cm.refresh();
|
||||||
|
}
|
||||||
|
CodeMirror.connect(window, "resize", function() {
|
||||||
|
var showing = document.body.getElementsByClassName("CodeMirror-fullscreen")[0];
|
||||||
|
if (!showing) return;
|
||||||
|
showing.CodeMirror.getScrollerElement().style.height = winHeight() + "px";
|
||||||
|
});
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
theme: "night",
|
||||||
|
extraKeys: {
|
||||||
|
"F11": function(cm) {
|
||||||
|
setFullScreen(cm, !isFullScreen(cm));
|
||||||
|
},
|
||||||
|
"Esc": function(cm) {
|
||||||
|
if (isFullScreen(cm)) setFullScreen(cm, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Press <strong>F11</strong> when cursor is in the editor to toggle full screen editing. <strong>Esc</strong> can also be used to <i>exit</i> full screen editing.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
40
codemirror-2.36/demo/loadmode.html
Normal file
40
codemirror-2.36/demo/loadmode.html
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Lazy Mode Loading Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/loadmode.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Lazy Mode Loading</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">This is the editor.
|
||||||
|
// It starts out in plain text mode,
|
||||||
|
# use the control below to load and apply a mode
|
||||||
|
"you'll see the highlighting of" this text /*change*/.
|
||||||
|
</textarea></form>
|
||||||
|
<p><input type=text value=javascript id=mode> <button type=button onclick="change()">change mode</button></p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.modeURL = "../mode/%N/%N.js";
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true
|
||||||
|
});
|
||||||
|
var modeInput = document.getElementById("mode");
|
||||||
|
CodeMirror.connect(modeInput, "keypress", function(e) {
|
||||||
|
if (e.keyCode == 13) change();
|
||||||
|
});
|
||||||
|
function change() {
|
||||||
|
editor.setOption("mode", modeInput.value);
|
||||||
|
CodeMirror.autoLoadMode(editor, modeInput.value);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
53
codemirror-2.36/demo/marker.html
Normal file
53
codemirror-2.36/demo/marker.html
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Breakpoint Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror-gutter {
|
||||||
|
width: 3em;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
.CodeMirror {
|
||||||
|
border: 1px solid #aaa;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Breakpoint demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
onGutterClick: function(cm, n) {
|
||||||
|
var info = cm.lineInfo(n);
|
||||||
|
if (info.markerText)
|
||||||
|
cm.clearMarker(n);
|
||||||
|
else
|
||||||
|
cm.setMarker(n, "<span style=\"color: #900\">●</span> %N%");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>Click the line-number gutter to add or remove 'breakpoints'.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
onGutterClick: function(cm, n) {
|
||||||
|
var info = cm.lineInfo(n);
|
||||||
|
if (info.markerText)
|
||||||
|
cm.clearMarker(n);
|
||||||
|
else
|
||||||
|
cm.setMarker(n, "<span style=\"color: #900\">●</span> %N%");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
38
codemirror-2.36/demo/matchhighlighter.html
Normal file
38
codemirror-2.36/demo/matchhighlighter.html
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Match Highlighter Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/searchcursor.js"></script>
|
||||||
|
<script src="../lib/util/match-highlighter.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}
|
||||||
|
|
||||||
|
span.CodeMirror-matchhighlight { background: #e9e9e9 }
|
||||||
|
.CodeMirror-focused span.CodeMirror-matchhighlight { background: #e7e4ff; !important }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Match Highlighter Demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">Select this text: hardToSpotVar
|
||||||
|
And everywhere else in your code where hardToSpotVar appears will automatically illuminate.
|
||||||
|
Give it a try! No more hardToSpotVars.</textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers : true,
|
||||||
|
onCursorActivity: function() {
|
||||||
|
editor.matchHighlight("CodeMirror-matchhighlight");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Highlight matches of selected text on select</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
60
codemirror-2.36/demo/multiplex.html
Normal file
60
codemirror-2.36/demo/multiplex.html
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Multiplexing Parser Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/multiplex.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border: 1px solid black;}
|
||||||
|
.cm-delimit {color: #fa4;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Multiplexing Parser Demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<html>
|
||||||
|
<body style="<<magic>>">
|
||||||
|
<h1><< this is not <html >></h1>
|
||||||
|
<<
|
||||||
|
multiline
|
||||||
|
not html
|
||||||
|
at all : &amp; <link/>
|
||||||
|
>>
|
||||||
|
<p>this is html again</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.defineMode("demo", function(config) {
|
||||||
|
return CodeMirror.multiplexingMode(
|
||||||
|
CodeMirror.getMode(config, "text/html"),
|
||||||
|
{open: "<<", close: ">>",
|
||||||
|
mode: CodeMirror.getMode(config, "text/plain"),
|
||||||
|
delimStyle: "delimit"}
|
||||||
|
// .. more multiplexed styles can follow here
|
||||||
|
);
|
||||||
|
});
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
mode: "demo",
|
||||||
|
lineNumbers: true,
|
||||||
|
lineWrapping: true
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Demonstration of a multiplexing mode, which, at certain
|
||||||
|
boundary strings, switches to one or more inner modes. The out
|
||||||
|
(HTML) mode does not get fed the content of the <code><<
|
||||||
|
>></code> blocks. See
|
||||||
|
the <a href="../doc/manual.html#util_multiplex">manual</a> and
|
||||||
|
the <a href="../lib/util/multiplex.js">source</a> for more
|
||||||
|
information.</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
59
codemirror-2.36/demo/mustache.html
Normal file
59
codemirror-2.36/demo/mustache.html
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Overlay Parser Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/overlay.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border: 1px solid black;}
|
||||||
|
.cm-mustache {color: #0ca;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Overlay Parser Demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<html>
|
||||||
|
<body>
|
||||||
|
<h1>{{title}}</h1>
|
||||||
|
<p>These are links to {{things}}:</p>
|
||||||
|
<ul>{{#links}}
|
||||||
|
<li><a href="{{url}}">{{text}}</a></li>
|
||||||
|
{{/links}}</ul>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.defineMode("mustache", function(config, parserConfig) {
|
||||||
|
var mustacheOverlay = {
|
||||||
|
token: function(stream, state) {
|
||||||
|
var ch;
|
||||||
|
if (stream.match("{{")) {
|
||||||
|
while ((ch = stream.next()) != null)
|
||||||
|
if (ch == "}" && stream.next() == "}") break;
|
||||||
|
stream.eat("}");
|
||||||
|
return "mustache";
|
||||||
|
}
|
||||||
|
while (stream.next() != null && !stream.match("{{", false)) {}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return CodeMirror.overlayMode(CodeMirror.getMode(config, parserConfig.backdrop || "text/html"), mustacheOverlay);
|
||||||
|
});
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {mode: "mustache"});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Demonstration of a mode that parses HTML, highlighting
|
||||||
|
the <a href="http://mustache.github.com/">Mustache</a> templating
|
||||||
|
directives inside of it by using the code
|
||||||
|
in <a href="../lib/util/overlay.js"><code>overlay.js</code></a>. View
|
||||||
|
source to see the 15 lines of code needed to accomplish this.</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
76
codemirror-2.36/demo/preview.html
Normal file
76
codemirror-2.36/demo/preview.html
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: HTML5 preview</title>
|
||||||
|
<script src=../lib/codemirror.js></script>
|
||||||
|
<script src=../mode/xml/xml.js></script>
|
||||||
|
<script src=../mode/javascript/javascript.js></script>
|
||||||
|
<script src=../mode/css/css.js></script>
|
||||||
|
<script src=../mode/htmlmixed/htmlmixed.js></script>
|
||||||
|
<link rel=stylesheet href=../lib/codemirror.css>
|
||||||
|
<link rel=stylesheet href=../doc/docs.css>
|
||||||
|
<style type=text/css>
|
||||||
|
.CodeMirror {
|
||||||
|
float: left;
|
||||||
|
width: 50%;
|
||||||
|
border: 1px solid black;
|
||||||
|
}
|
||||||
|
iframe {
|
||||||
|
width: 49%;
|
||||||
|
float: left;
|
||||||
|
height: 300px;
|
||||||
|
border: 1px solid black;
|
||||||
|
border-left: 0px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: HTML5 preview</h1>
|
||||||
|
<textarea id=code name=code>
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset=utf-8>
|
||||||
|
<title>HTML5 canvas demo</title>
|
||||||
|
<style>p {font-family: monospace;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p>Canvas pane goes here:</p>
|
||||||
|
<canvas id=pane width=300 height=200></canvas>
|
||||||
|
<script>
|
||||||
|
var canvas = document.getElementById('pane');
|
||||||
|
var context = canvas.getContext('2d');
|
||||||
|
|
||||||
|
context.fillStyle = 'rgb(250,0,0)';
|
||||||
|
context.fillRect(10, 10, 55, 50);
|
||||||
|
|
||||||
|
context.fillStyle = 'rgba(0, 0, 250, 0.5)';
|
||||||
|
context.fillRect(30, 30, 55, 50);
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html></textarea>
|
||||||
|
<iframe id=preview></iframe>
|
||||||
|
<script>
|
||||||
|
var delay;
|
||||||
|
// Initialize CodeMirror editor with a nice html5 canvas demo.
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById('code'), {
|
||||||
|
mode: 'text/html',
|
||||||
|
tabMode: 'indent',
|
||||||
|
onChange: function() {
|
||||||
|
clearTimeout(delay);
|
||||||
|
delay = setTimeout(updatePreview, 300);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function updatePreview() {
|
||||||
|
var previewFrame = document.getElementById('preview');
|
||||||
|
var preview = previewFrame.contentDocument || previewFrame.contentWindow.document;
|
||||||
|
preview.open();
|
||||||
|
preview.write(editor.getValue());
|
||||||
|
preview.close();
|
||||||
|
}
|
||||||
|
setTimeout(updatePreview, 300);
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
42
codemirror-2.36/demo/resize.html
Normal file
42
codemirror-2.36/demo/resize.html
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Autoresize Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/css/css.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {
|
||||||
|
border: 1px solid #eee;
|
||||||
|
}
|
||||||
|
.CodeMirror-scroll {
|
||||||
|
height: auto;
|
||||||
|
overflow-y: hidden;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Autoresize demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
.CodeMirror-scroll {
|
||||||
|
height: auto;
|
||||||
|
overflow-y: hidden;
|
||||||
|
overflow-x: auto;
|
||||||
|
}</textarea></form>
|
||||||
|
|
||||||
|
<p>By setting a few CSS properties, CodeMirror can be made to
|
||||||
|
automatically resize to fit its content.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
50
codemirror-2.36/demo/runmode.html
Normal file
50
codemirror-2.36/demo/runmode.html
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Mode Runner Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/runmode.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Mode Runner Demo</h1>
|
||||||
|
|
||||||
|
<textarea id="code" style="width: 90%; height: 7em; border: 1px solid black; padding: .2em .4em;">
|
||||||
|
<foobar>
|
||||||
|
<blah>Enter your xml here and press the button below to display
|
||||||
|
it as highlighted by the CodeMirror XML mode</blah>
|
||||||
|
<tag2 foo="2" bar="&quot;bar&quot;"/>
|
||||||
|
</foobar></textarea><br>
|
||||||
|
<button onclick="doHighlight();">Highlight!</button>
|
||||||
|
<pre id="output" class="cm-s-default"></pre>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function doHighlight() {
|
||||||
|
CodeMirror.runMode(document.getElementById("code").value, "application/xml",
|
||||||
|
document.getElementById("output"));
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Running a CodeMirror mode outside of the editor.
|
||||||
|
The <code>CodeMirror.runMode</code> function, defined
|
||||||
|
in <code><a href="../lib/util/runmode.js">lib/runmode.js</a></code> takes the following arguments:</p>
|
||||||
|
|
||||||
|
<dl>
|
||||||
|
<dt><code>text (string)</code></dt>
|
||||||
|
<dd>The document to run through the highlighter.</dd>
|
||||||
|
<dt><code>mode (<a href="../doc/manual.html#option_mode">mode spec</a>)</code></dt>
|
||||||
|
<dd>The mode to use (must be loaded as normal).</dd>
|
||||||
|
<dt><code>output (function or DOM node)</code></dt>
|
||||||
|
<dd>If this is a function, it will be called for each token with
|
||||||
|
two arguments, the token's text and the token's style class (may
|
||||||
|
be <code>null</code> for unstyled tokens). If it is a DOM node,
|
||||||
|
the tokens will be converted to <code>span</code> elements as in
|
||||||
|
an editor, and inserted into the node
|
||||||
|
(through <code>innerHTML</code>).</dd>
|
||||||
|
</dl>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
85
codemirror-2.36/demo/search.html
Normal file
85
codemirror-2.36/demo/search.html
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Search/Replace Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<script src="../lib/util/dialog.js"></script>
|
||||||
|
<link rel="stylesheet" href="../lib/util/dialog.css">
|
||||||
|
<script src="../lib/util/searchcursor.js"></script>
|
||||||
|
<script src="../lib/util/search.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}
|
||||||
|
dt {font-family: monospace; color: #666;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Search/Replace Demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<dt id="option_indentWithTabs"><code>indentWithTabs (boolean)</code></dt>
|
||||||
|
<dd>Whether, when indenting, the first N*8 spaces should be
|
||||||
|
replaced by N tabs. Default is false.</dd>
|
||||||
|
|
||||||
|
<dt id="option_tabMode"><code>tabMode (string)</code></dt>
|
||||||
|
<dd>Determines what happens when the user presses the tab key.
|
||||||
|
Must be one of the following:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"classic" (the default)</code></dt>
|
||||||
|
<dd>When nothing is selected, insert a tab. Otherwise,
|
||||||
|
behave like the <code>"shift"</code> mode. (When shift is
|
||||||
|
held, this behaves like the <code>"indent"</code> mode.)</dd>
|
||||||
|
<dt><code>"shift"</code></dt>
|
||||||
|
<dd>Indent all selected lines by
|
||||||
|
one <a href="#option_indentUnit"><code>indentUnit</code></a>.
|
||||||
|
If shift was held while pressing tab, un-indent all selected
|
||||||
|
lines one unit.</dd>
|
||||||
|
<dt><code>"indent"</code></dt>
|
||||||
|
<dd>Indent the line the 'correctly', based on its syntactic
|
||||||
|
context. Only works if the
|
||||||
|
mode <a href="#indent">supports</a> it.</dd>
|
||||||
|
<dt><code>"default"</code></dt>
|
||||||
|
<dd>Do not capture tab presses, let the browser apply its
|
||||||
|
default behaviour (which usually means it skips to the next
|
||||||
|
control).</dd>
|
||||||
|
</dl></dd>
|
||||||
|
|
||||||
|
<dt id="option_enterMode"><code>enterMode (string)</code></dt>
|
||||||
|
<dd>Determines whether and how new lines are indented when the
|
||||||
|
enter key is pressed. The following modes are supported:
|
||||||
|
<dl>
|
||||||
|
<dt><code>"indent" (the default)</code></dt>
|
||||||
|
<dd>Use the mode's indentation rules to give the new line
|
||||||
|
the correct indentation.</dd>
|
||||||
|
<dt><code>"keep"</code></dt>
|
||||||
|
<dd>Indent the line the same as the previous line.</dd>
|
||||||
|
<dt><code>"flat"</code></dt>
|
||||||
|
<dd>Do not indent the new line.</dd>
|
||||||
|
</dl></dd>
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {mode: "text/html", lineNumbers: true});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Demonstration of primitive search/replace functionality. The
|
||||||
|
keybindings (which can be overridden by custom keymaps) are:</p>
|
||||||
|
<dl>
|
||||||
|
<dt>Ctrl-F / Cmd-F</dt><dd>Start searching</dd>
|
||||||
|
<dt>Ctrl-G / Cmd-G</dt><dd>Find next</dd>
|
||||||
|
<dt>Shift-Ctrl-G / Shift-Cmd-G</dt><dd>Find previous</dd>
|
||||||
|
<dt>Shift-Ctrl-F / Cmd-Option-F</dt><dd>Replace</dd>
|
||||||
|
<dt>Shift-Ctrl-R / Shift-Cmd-Option-F</dt><dd>Replace all</dd>
|
||||||
|
</dl>
|
||||||
|
<p>Searching is enabled by
|
||||||
|
including <a href="../lib/util/search.js">lib/util/search.js</a>
|
||||||
|
and <a href="../lib/util/searchcursor.js">lib/util/searchcursor.js</a>.
|
||||||
|
For good-looking input dialogs, you also want to include
|
||||||
|
<a href="../lib/util/dialog.js">lib/util/dialog.js</a>
|
||||||
|
and <a href="../lib/util/dialog.css">lib/util/dialog.css</a>.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
81
codemirror-2.36/demo/theme.html
Normal file
81
codemirror-2.36/demo/theme.html
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Theme Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<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/ambiance.css">
|
||||||
|
<link rel="stylesheet" href="../theme/blackboard.css">
|
||||||
|
<link rel="stylesheet" href="../theme/vibrant-ink.css">
|
||||||
|
<link rel="stylesheet" href="../theme/twilight.css">
|
||||||
|
<script src="../mode/javascript/javascript.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border: 1px solid black; font-size:13px}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Theme demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
function findSequence(goal) {
|
||||||
|
function find(start, history) {
|
||||||
|
if (start == goal)
|
||||||
|
return history;
|
||||||
|
else if (start > goal)
|
||||||
|
return null;
|
||||||
|
else
|
||||||
|
return find(start + 5, "(" + history + " + 5)") ||
|
||||||
|
find(start * 3, "(" + history + " * 3)");
|
||||||
|
}
|
||||||
|
return find(1, "1");
|
||||||
|
}</textarea></form>
|
||||||
|
|
||||||
|
<p>Select a theme: <select onchange="selectTheme()" id=select>
|
||||||
|
<option selected>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>monokai</option>
|
||||||
|
<option>neat</option>
|
||||||
|
<option>night</option>
|
||||||
|
<option>rubyblue</option>
|
||||||
|
<option>twilight</option>
|
||||||
|
<option>vibrant-ink</option>
|
||||||
|
<option>xq-dark</option>
|
||||||
|
</select>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true
|
||||||
|
});
|
||||||
|
var input = document.getElementById("select");
|
||||||
|
function selectTheme() {
|
||||||
|
var theme = input.options[input.selectedIndex].innerHTML;
|
||||||
|
editor.setOption("theme", theme);
|
||||||
|
}
|
||||||
|
var choice = document.location.search && document.location.search.slice(1);
|
||||||
|
if (choice) {
|
||||||
|
input.value = choice;
|
||||||
|
editor.setOption("theme", choice);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
53
codemirror-2.36/demo/vim.html
Normal file
53
codemirror-2.36/demo/vim.html
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Vim bindings demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/clike/clike.js"></script>
|
||||||
|
<script src="../keymap/vim.js"></script>
|
||||||
|
<script src="../lib/util/dialog.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
<link rel="stylesheet" href="../lib/util/dialog.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid #eee; border-bottom: 1px solid #eee;}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Vim bindings demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
#include "syscalls.h"
|
||||||
|
/* getchar: simple buffered version */
|
||||||
|
int getchar(void)
|
||||||
|
{
|
||||||
|
static char buf[BUFSIZ];
|
||||||
|
static char *bufp = buf;
|
||||||
|
static int n = 0;
|
||||||
|
if (n == 0) { /* buffer is empty */
|
||||||
|
n = read(0, buf, sizeof buf);
|
||||||
|
bufp = buf;
|
||||||
|
}
|
||||||
|
return (--n >= 0) ? (unsigned char) *bufp++ : EOF;
|
||||||
|
}
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>The vim keybindings are enabled by
|
||||||
|
including <a href="../keymap/vim.js">keymap/vim.js</a> and setting
|
||||||
|
the <code>keyMap</code> option to <code>"vim"</code>. Because
|
||||||
|
CodeMirror's internal API is quite different from Vim, they are only
|
||||||
|
a loose approximation of actual vim bindings, though.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.commands.save = function(){ alert("Saving"); };
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
mode: "text/x-csrc",
|
||||||
|
keyMap: "vim"
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
53
codemirror-2.36/demo/visibletabs.html
Normal file
53
codemirror-2.36/demo/visibletabs.html
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Visible tabs demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../mode/clike/clike.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.CodeMirror {border-top: 1px solid #eee; border-bottom: 1px solid #eee;}
|
||||||
|
.cm-tab {
|
||||||
|
background: url();
|
||||||
|
background-position: right;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Visible tabs demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
#include "syscalls.h"
|
||||||
|
/* getchar: simple buffered version */
|
||||||
|
int getchar(void)
|
||||||
|
{
|
||||||
|
static char buf[BUFSIZ];
|
||||||
|
static char *bufp = buf;
|
||||||
|
static int n = 0;
|
||||||
|
if (n == 0) { /* buffer is empty */
|
||||||
|
n = read(0, buf, sizeof buf);
|
||||||
|
bufp = buf;
|
||||||
|
}
|
||||||
|
return (--n >= 0) ? (unsigned char) *bufp++ : EOF;
|
||||||
|
}
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<p>Tabs inside the editor are spans with the
|
||||||
|
class <code>cm-tab</code>, and can be styled.
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
tabSize: 4,
|
||||||
|
indentUnit: 4,
|
||||||
|
indentWithTabs: true,
|
||||||
|
mode: "text/x-csrc"
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
71
codemirror-2.36/demo/xmlcomplete.html
Normal file
71
codemirror-2.36/demo/xmlcomplete.html
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: XML Autocomplete Demo</title>
|
||||||
|
<link rel="stylesheet" href="../lib/codemirror.css">
|
||||||
|
<script src="../lib/codemirror.js"></script>
|
||||||
|
<script src="../lib/util/simple-hint.js"></script>
|
||||||
|
<link rel="stylesheet" href="../lib/util/simple-hint.css">
|
||||||
|
<script src="../lib/util/closetag.js"></script>
|
||||||
|
<script src="../lib/util/xml-hint.js"></script>
|
||||||
|
<script src="../mode/xml/xml.js"></script>
|
||||||
|
<link rel="stylesheet" href="../doc/docs.css">
|
||||||
|
<style type="text/css">.CodeMirror {border: 1px solid #eee;} .CodeMirror-scroll { height: 100% }</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: XML Autocomplete demo</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code"></textarea></form>
|
||||||
|
|
||||||
|
<p>Type '<' or space inside tag or
|
||||||
|
press <strong>ctrl-space</strong> to activate autocompletion. See
|
||||||
|
the code (<a href="../lib/util/simple-hint.js">here</a>
|
||||||
|
and <a href="../lib/util/xml-hint.js">here</a>) to figure out how
|
||||||
|
it works.</p>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
CodeMirror.xmlHints['<'] = [
|
||||||
|
'levelTop',
|
||||||
|
'levelRoot',
|
||||||
|
'mainLevel'
|
||||||
|
];
|
||||||
|
|
||||||
|
CodeMirror.xmlHints['<levelTop '] =
|
||||||
|
CodeMirror.xmlHints['<levelRoot '] =
|
||||||
|
CodeMirror.xmlHints['<mainLevel '] = [
|
||||||
|
'property1111',
|
||||||
|
'property2222'
|
||||||
|
];
|
||||||
|
|
||||||
|
CodeMirror.xmlHints['<levelTop><'] =
|
||||||
|
CodeMirror.xmlHints['<levelRoot><'] =
|
||||||
|
CodeMirror.xmlHints['<mainLevel><'] = [
|
||||||
|
'second',
|
||||||
|
'two'
|
||||||
|
];
|
||||||
|
|
||||||
|
CodeMirror.xmlHints['<levelTop><second '] = [
|
||||||
|
'secondProperty'
|
||||||
|
];
|
||||||
|
|
||||||
|
CodeMirror.xmlHints['<levelTop><second><'] = [
|
||||||
|
'three',
|
||||||
|
'x-three'
|
||||||
|
];
|
||||||
|
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
value: '',
|
||||||
|
mode: 'text/html',
|
||||||
|
lineNumbers: true,
|
||||||
|
extraKeys: {
|
||||||
|
"'>'": function(cm) { cm.closeTag(cm, '>'); },
|
||||||
|
"'/'": function(cm) { cm.closeTag(cm, '/'); },
|
||||||
|
"' '": function(cm) { CodeMirror.xmlHint(cm, ' '); },
|
||||||
|
"'<'": function(cm) { CodeMirror.xmlHint(cm, '<'); },
|
||||||
|
"Ctrl-Space": function(cm) { CodeMirror.xmlHint(cm, ''); }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
BIN
codemirror-2.36/doc/baboon.png
Normal file
BIN
codemirror-2.36/doc/baboon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 23 KiB |
153
codemirror-2.36/doc/baboon_vector.svg
Normal file
153
codemirror-2.36/doc/baboon_vector.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 17 KiB |
168
codemirror-2.36/doc/compress.html
Normal file
168
codemirror-2.36/doc/compress.html
Normal file
|
@ -0,0 +1,168 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror: Compression Helper</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* Script compression
|
||||||
|
helper */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p>To optimize loading CodeMirror, especially when including a
|
||||||
|
bunch of different modes, it is recommended that you combine and
|
||||||
|
minify (and preferably also gzip) the scripts. This page makes
|
||||||
|
those first two steps very easy. Simply select the version and
|
||||||
|
scripts you need in the form below, and
|
||||||
|
click <strong>Compress</strong> to download the minified script
|
||||||
|
file.</p>
|
||||||
|
|
||||||
|
<form id="form" action="http://marijnhaverbeke.nl/uglifyjs" method="post">
|
||||||
|
<input type="hidden" id="download" name="download" value="codemirror-compressed.js"/>
|
||||||
|
<p>Version: <select id="version" onchange="setVersion(this);" style="padding: 1px">
|
||||||
|
<option value="http://codemirror.net/">HEAD</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v3.0rc1;f=">3.0rc1</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v3.0beta2;f=">3.0beta2</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v3.0beta1;f=">3.0beta1</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.36;f=">2.36</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.35;f=">2.35</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.34;f=">2.34</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.33;f=">2.33</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.32;f=">2.32</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.31;f=">2.31</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.3;f=">2.3</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.25;f=">2.25</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.24;f=">2.24</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.23;f=">2.23</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.22;f=">2.22</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.21;f=">2.21</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.2;f=">2.2</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.18;f=">2.18</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.16;f=">2.16</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.15;f=">2.15</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.13;f=">2.13</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.12;f=">2.12</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.11;f=">2.11</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.1;f=">2.1</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.02;f=">2.02</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.01;f=">2.01</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=v2.0;f=">2.0</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=beta2;f=">beta2</option>
|
||||||
|
<option value="http://marijnhaverbeke.nl/git/codemirror?a=blob_plain;hb=beta1;f=">beta1</option>
|
||||||
|
</select></p>
|
||||||
|
|
||||||
|
<select multiple="multiple" size="20" name="code_url" style="width: 40em;" class="field" id="files">
|
||||||
|
<optgroup label="CodeMirror Library">
|
||||||
|
<option value="http://codemirror.net/lib/codemirror.js" selected>codemirror.js</option>
|
||||||
|
</optgroup>
|
||||||
|
<optgroup label="Modes">
|
||||||
|
<option value="http://codemirror.net/mode/clike/clike.js">clike.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/clojure/clojure.js">clojure.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/coffeescript/coffeescript.js">coffeescript.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/commonlisp/commonlisp.js">commonlisp.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/css/css.js">css.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/diff/diff.js">diff.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/ecl/ecl.js">ecl.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/erlang/erlang.js">erlang.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/gfm/gfm.js">gfm.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/go/go.js">go.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/groovy/groovy.js">groovy.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/haskell/haskell.js">haskell.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/haxe/haxe.js">haxe.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/htmlembedded/htmlembedded.js">htmlembedded.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/htmlmixed/htmlmixed.js">htmlmixed.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/javascript/javascript.js">javascript.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/jinja2/jinja2.js">jinja2.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/less/less.js">less.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/lua/lua.js">lua.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/markdown/markdown.js">markdown.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/mysql/mysql.js">mysql.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/ntriples/ntriples.js">ntriples.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/ocaml/ocaml.js">ocaml.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/pascal/pascal.js">pascal.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/perl/perl.js">perl.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/php/php.js">php.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/pig/pig.js">pig.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/plsql/plsql.js">plsql.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/properties/properties.js">properties.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/python/python.js">python.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/r/r.js">r.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/rpm/changes/changes.js">rpm/changes.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/rpm/spec/spec.js">rpm/spec.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/rst/rst.js">rst.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/ruby/ruby.js">ruby.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/rust/rust.js">rust.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/scheme/scheme.js">scheme.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/shell/shell.js">shell.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/sieve/sieve.js">sieve.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/smalltalk/smalltalk.js">smalltalk.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/smarty/smarty.js">smarty.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/sparql/sparql.js">sparql.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/stex/stex.js">stex.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/tiddlywiki/tiddlywiki.js">tiddlywiki.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/tiki/tiki.js">tiki.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/vb/vb.js">vb.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/vbscript/vbscript.js">vbscript.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/velocity/velocity.js">velocity.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/verilog/verilog.js">verilog.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/xml/xml.js">xml.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/xquery/xquery.js">xquery.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/yaml/yaml.js">yaml.js</option>
|
||||||
|
<option value="http://codemirror.net/mode/z80/z80.js">z80.js</option>
|
||||||
|
</optgroup>
|
||||||
|
<optgroup label="Utilities and add-ons">
|
||||||
|
<option value="http://codemirror.net/lib/util/overlay.js">overlay.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/multiplex.js">multiplex.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/runmode.js">runmode.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/simple-hint.js">simple-hint.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/javascript-hint.js">javascript-hint.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/xml-hint.js">xml-hint.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/foldcode.js">foldcode.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/dialog.js">dialog.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/search.js">search.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/searchcursor.js">searchcursor.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/formatting.js">formatting.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/match-highlighter.js">match-highlighter.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/closetag.js">closetag.js</option>
|
||||||
|
<option value="http://codemirror.net/lib/util/loadmode.js">loadmode.js</option>
|
||||||
|
</optgroup>
|
||||||
|
<optgroup label="Keymaps">
|
||||||
|
<option value="http://codemirror.net/keymap/emacs.js">emacs.js</option>
|
||||||
|
<option value="http://codemirror.net/keymap/vim.js">vim.js</option>
|
||||||
|
</optgroup>
|
||||||
|
</select></p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<button type="submit">Compress</button> with <a href="http://github.com/mishoo/UglifyJS/">UglifyJS</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>Custom code to add to the compressed file:<textarea name="js_code" style="width: 100%; height: 15em;" class="field"></textarea></p>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
function setVersion(ver) {
|
||||||
|
var urlprefix = ver.options[ver.selectedIndex].value;
|
||||||
|
var select = document.getElementById("files"), m;
|
||||||
|
for (var optgr = select.firstChild; optgr; optgr = optgr.nextSibling)
|
||||||
|
for (var opt = optgr.firstChild; opt; opt = opt.nextSibling) {
|
||||||
|
if (opt.nodeName != "OPTION")
|
||||||
|
continue;
|
||||||
|
else if (m = opt.value.match(/^http:\/\/codemirror.net\/(.*)$/))
|
||||||
|
opt.value = urlprefix + m[1];
|
||||||
|
else if (m = opt.value.match(/http:\/\/marijnhaverbeke.nl\/git\/codemirror\?a=blob_plain;hb=[^;]+;f=(.*)$/))
|
||||||
|
opt.value = urlprefix + m[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
167
codemirror-2.36/doc/docs.css
Normal file
167
codemirror-2.36/doc/docs.css
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
body {
|
||||||
|
font-family: Droid Sans, Arial, sans-serif;
|
||||||
|
line-height: 1.5;
|
||||||
|
max-width: 64.3em;
|
||||||
|
margin: 3em auto;
|
||||||
|
padding: 0 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
letter-spacing: -3px;
|
||||||
|
font-size: 3.23em;
|
||||||
|
font-weight: bold;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-size: 1.23em;
|
||||||
|
font-weight: bold;
|
||||||
|
margin: .5em 0;
|
||||||
|
letter-spacing: -1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
font-size: 1em;
|
||||||
|
font-weight: bold;
|
||||||
|
margin: .4em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
background-color: #eee;
|
||||||
|
-moz-border-radius: 6px;
|
||||||
|
-webkit-border-radius: 6px;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code {
|
||||||
|
margin: 0 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grey {
|
||||||
|
background-color: #eee;
|
||||||
|
border-radius: 6px;
|
||||||
|
margin-bottom: 1.65em;
|
||||||
|
margin-top: 0.825em;
|
||||||
|
padding: 0.825em 1.65em;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
img.logo {
|
||||||
|
position: absolute;
|
||||||
|
right: -1em;
|
||||||
|
bottom: 4px;
|
||||||
|
max-width: 23.6875em; /* Scale image down with text to prevent clipping */
|
||||||
|
}
|
||||||
|
|
||||||
|
.grey > pre {
|
||||||
|
background:none;
|
||||||
|
border-radius:0;
|
||||||
|
padding:0;
|
||||||
|
margin:0;
|
||||||
|
font-size:2.2em;
|
||||||
|
line-height:1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:link, a:visited, .quasilink {
|
||||||
|
color: #df0019;
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover, .quasilink:hover {
|
||||||
|
color: #800004;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 a:link, h1 a:visited, h1 a:hover {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
margin: 0;
|
||||||
|
padding-left: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.download {
|
||||||
|
color: white;
|
||||||
|
background-color: #df0019;
|
||||||
|
width: 100%;
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
font-size: 1.23em;
|
||||||
|
font-weight: bold;
|
||||||
|
text-decoration: none;
|
||||||
|
-moz-border-radius: 6px;
|
||||||
|
-webkit-border-radius: 6px;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: .5em 0;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.download:hover {
|
||||||
|
background-color: #bb0010;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rel {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rel-note {
|
||||||
|
color: #777;
|
||||||
|
font-size: .9em;
|
||||||
|
margin-top: .1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo-braces {
|
||||||
|
color: #df0019;
|
||||||
|
position: relative;
|
||||||
|
top: -4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blk {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left {
|
||||||
|
margin-right: 20.68em;
|
||||||
|
max-width: 37em;
|
||||||
|
padding-right: 6.53em;
|
||||||
|
padding-bottom: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left1 {
|
||||||
|
width: 15.24em;
|
||||||
|
padding-right: 6.45em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left2 {
|
||||||
|
max-width: 15.24em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.right {
|
||||||
|
width: 20.68em;
|
||||||
|
margin-left: -20.68em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.leftbig {
|
||||||
|
width: 42.44em;
|
||||||
|
padding-right: 6.53em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rightsmall {
|
||||||
|
width: 15.24em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear:after {
|
||||||
|
visibility: hidden;
|
||||||
|
display: block;
|
||||||
|
font-size: 0;
|
||||||
|
content: " ";
|
||||||
|
clear: both;
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
.clear { display: inline-block; }
|
||||||
|
/* start commented backslash hack \*/
|
||||||
|
* html .clear { height: 1%; }
|
||||||
|
.clear { display: block; }
|
||||||
|
/* close commented backslash hack */
|
497
codemirror-2.36/doc/internals.html
Normal file
497
codemirror-2.36/doc/internals.html
Normal file
|
@ -0,0 +1,497 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror: Internals</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
<style>dl dl {margin: 0;} .update {color: #d40 !important}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* (Re-) Implementing A Syntax-
|
||||||
|
Highlighting Editor in JavaScript */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="clear"><div class="leftbig blk">
|
||||||
|
|
||||||
|
<p style="font-size: 85%" id="intro">
|
||||||
|
<strong>Topic:</strong> JavaScript, code editor implementation<br>
|
||||||
|
<strong>Author:</strong> Marijn Haverbeke<br>
|
||||||
|
<strong>Date:</strong> March 2nd 2011 (updated November 13th 2011)
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>This is a followup to
|
||||||
|
my <a href="http://codemirror.net/story.html">Brutal Odyssey to the
|
||||||
|
Dark Side of the DOM Tree</a> story. That one describes the
|
||||||
|
mind-bending process of implementing (what would become) CodeMirror 1.
|
||||||
|
This one describes the internals of CodeMirror 2, a complete rewrite
|
||||||
|
and rethink of the old code base. I wanted to give this piece another
|
||||||
|
Hunter Thompson copycat subtitle, but somehow that would be out of
|
||||||
|
place—the process this time around was one of straightforward
|
||||||
|
engineering, requiring no serious mind-bending whatsoever.</p>
|
||||||
|
|
||||||
|
<p>So, what is wrong with CodeMirror 1? I'd estimate, by mailing list
|
||||||
|
activity and general search-engine presence, that it has been
|
||||||
|
integrated into about a thousand systems by now. The most prominent
|
||||||
|
one, since a few weeks,
|
||||||
|
being <a href="http://googlecode.blogspot.com/2011/01/make-quick-fixes-quicker-on-google.html">Google
|
||||||
|
code's project hosting</a>. It works, and it's being used widely.</a>
|
||||||
|
|
||||||
|
<p>Still, I did not start replacing it because I was bored. CodeMirror
|
||||||
|
1 was heavily reliant on <code>designMode</code>
|
||||||
|
or <code>contentEditable</code> (depending on the browser). Neither of
|
||||||
|
these are well specified (HTML5 tries
|
||||||
|
to <a href="http://www.w3.org/TR/html5/editing.html#contenteditable">specify</a>
|
||||||
|
their basics), and, more importantly, they tend to be one of the more
|
||||||
|
obscure and buggy areas of browser functionality—CodeMirror, by using
|
||||||
|
this functionality in a non-typical way, was constantly running up
|
||||||
|
against browser bugs. WebKit wouldn't show an empty line at the end of
|
||||||
|
the document, and in some releases would suddenly get unbearably slow.
|
||||||
|
Firefox would show the cursor in the wrong place. Internet Explorer
|
||||||
|
would insist on linkifying everything that looked like a URL or email
|
||||||
|
address, a behaviour that can't be turned off. Some bugs I managed to
|
||||||
|
work around (which was often a frustrating, painful process), others,
|
||||||
|
such as the Firefox cursor placement, I gave up on, and had to tell
|
||||||
|
user after user that they were known problems, but not something I
|
||||||
|
could help.</p>
|
||||||
|
|
||||||
|
<p>Also, there is the fact that <code>designMode</code> (which seemed
|
||||||
|
to be less buggy than <code>contentEditable</code> in Webkit and
|
||||||
|
Firefox, and was thus used by CodeMirror 1 in those browsers) requires
|
||||||
|
a frame. Frames are another tricky area. It takes some effort to
|
||||||
|
prevent getting tripped up by domain restrictions, they don't
|
||||||
|
initialize synchronously, behave strangely in response to the back
|
||||||
|
button, and, on several browsers, can't be moved around the DOM
|
||||||
|
without having them re-initialize. They did provide a very nice way to
|
||||||
|
namespace the library, though—CodeMirror 1 could freely pollute the
|
||||||
|
namespace inside the frame.</p>
|
||||||
|
|
||||||
|
<p>Finally, working with an editable document means working with
|
||||||
|
selection in arbitrary DOM structures. Internet Explorer (8 and
|
||||||
|
before) has an utterly different (and awkward) selection API than all
|
||||||
|
of the other browsers, and even among the different implementations of
|
||||||
|
<code>document.selection</code>, details about how exactly a selection
|
||||||
|
is represented vary quite a bit. Add to that the fact that Opera's
|
||||||
|
selection support tended to be very buggy until recently, and you can
|
||||||
|
imagine why CodeMirror 1 contains 700 lines of selection-handling
|
||||||
|
code.</p>
|
||||||
|
|
||||||
|
<p>And that brings us to the main issue with the CodeMirror 1
|
||||||
|
code base: The proportion of browser-bug-workarounds to real
|
||||||
|
application code was getting dangerously high. By building on top of a
|
||||||
|
few dodgy features, I put the system in a vulnerable position—any
|
||||||
|
incompatibility and bugginess in these features, I had to paper over
|
||||||
|
with my own code. Not only did I have to do some serious stunt-work to
|
||||||
|
get it to work on older browsers (as detailed in the
|
||||||
|
previous <a href="http://codemirror.net/story.html">story</a>), things
|
||||||
|
also kept breaking in newly released versions, requiring me to come up
|
||||||
|
with <em>new</em> scary hacks in order to keep up. This was starting
|
||||||
|
to lose its appeal.</p>
|
||||||
|
|
||||||
|
<h2 id="approach">General Approach</h2>
|
||||||
|
|
||||||
|
<p>What CodeMirror 2 does is try to sidestep most of the hairy hacks
|
||||||
|
that came up in version 1. I owe a lot to the
|
||||||
|
<a href="http://ace.ajax.org">ACE</a> editor for inspiration on how to
|
||||||
|
approach this.</p>
|
||||||
|
|
||||||
|
<p>I absolutely did not want to be completely reliant on key events to
|
||||||
|
generate my input. Every JavaScript programmer knows that key event
|
||||||
|
information is horrible and incomplete. Some people (most awesomely
|
||||||
|
Mihai Bazon with <a href="http://ymacs.org">Ymacs</a>) have been able
|
||||||
|
to build more or less functioning editors by directly reading key
|
||||||
|
events, but it takes a lot of work (the kind of never-ending, fragile
|
||||||
|
work I described earlier), and will never be able to properly support
|
||||||
|
things like multi-keystoke international character
|
||||||
|
input. <a href="#keymap" class="update">[see below for caveat]</a></p>
|
||||||
|
|
||||||
|
<p>So what I do is focus a hidden textarea, and let the browser
|
||||||
|
believe that the user is typing into that. What we show to the user is
|
||||||
|
a DOM structure we built to represent his document. If this is updated
|
||||||
|
quickly enough, and shows some kind of believable cursor, it feels
|
||||||
|
like a real text-input control.</p>
|
||||||
|
|
||||||
|
<p>Another big win is that this DOM representation does not have to
|
||||||
|
span the whole document. Some CodeMirror 1 users insisted that they
|
||||||
|
needed to put a 30 thousand line XML document into CodeMirror. Putting
|
||||||
|
all that into the DOM takes a while, especially since, for some
|
||||||
|
reason, an editable DOM tree is slower than a normal one on most
|
||||||
|
browsers. If we have full control over what we show, we must only
|
||||||
|
ensure that the visible part of the document has been added, and can
|
||||||
|
do the rest only when needed. (Fortunately, the <code>onscroll</code>
|
||||||
|
event works almost the same on all browsers, and lends itself well to
|
||||||
|
displaying things only as they are scrolled into view.)</p>
|
||||||
|
|
||||||
|
<h2 id="input">Input</h2>
|
||||||
|
|
||||||
|
<p>ACE uses its hidden textarea only as a text input shim, and does
|
||||||
|
all cursor movement and things like text deletion itself by directly
|
||||||
|
handling key events. CodeMirror's way is to let the browser do its
|
||||||
|
thing as much as possible, and not, for example, define its own set of
|
||||||
|
key bindings. One way to do this would have been to have the whole
|
||||||
|
document inside the hidden textarea, and after each key event update
|
||||||
|
the display DOM to reflect what's in that textarea.</p>
|
||||||
|
|
||||||
|
<p>That'd be simple, but it is not realistic. For even medium-sized
|
||||||
|
document the editor would be constantly munging huge strings, and get
|
||||||
|
terribly slow. What CodeMirror 2 does is put the current selection,
|
||||||
|
along with an extra line on the top and on the bottom, into the
|
||||||
|
textarea.</p>
|
||||||
|
|
||||||
|
<p>This means that the arrow keys (and their ctrl-variations), home,
|
||||||
|
end, etcetera, do not have to be handled specially. We just read the
|
||||||
|
cursor position in the textarea, and update our cursor to match it.
|
||||||
|
Also, copy and paste work pretty much for free, and people get their
|
||||||
|
native key bindings, without any special work on my part. For example,
|
||||||
|
I have emacs key bindings configured for Chrome and Firefox. There is
|
||||||
|
no way for a script to detect this. <a class="update"
|
||||||
|
href="#keymap">[no longer the case]</a></p>
|
||||||
|
|
||||||
|
<p>Of course, since only a small part of the document sits in the
|
||||||
|
textarea, keys like page up and ctrl-end won't do the right thing.
|
||||||
|
CodeMirror is catching those events and handling them itself.</p>
|
||||||
|
|
||||||
|
<h2 id="selection">Selection</h2>
|
||||||
|
|
||||||
|
<p>Getting and setting the selection range of a textarea in modern
|
||||||
|
browsers is trivial—you just use the <code>selectionStart</code>
|
||||||
|
and <code>selectionEnd</code> properties. On IE you have to do some
|
||||||
|
insane stuff with temporary ranges and compensating for the fact that
|
||||||
|
moving the selection by a 'character' will treat \r\n as a single
|
||||||
|
character, but even there it is possible to build functions that
|
||||||
|
reliably set and get the selection range.</p>
|
||||||
|
|
||||||
|
<p>But consider this typical case: When I'm somewhere in my document,
|
||||||
|
press shift, and press the up arrow, something gets selected. Then, if
|
||||||
|
I, still holding shift, press the up arrow again, the top of my
|
||||||
|
selection is adjusted. The selection remembers where its <em>head</em>
|
||||||
|
and its <em>anchor</em> are, and moves the head when we shift-move.
|
||||||
|
This is a generally accepted property of selections, and done right by
|
||||||
|
every editing component built in the past twenty years.</p>
|
||||||
|
|
||||||
|
<p>But not something that the browser selection APIs expose.</p>
|
||||||
|
|
||||||
|
<p>Great. So when someone creates an 'upside-down' selection, the next
|
||||||
|
time CodeMirror has to update the textarea, it'll re-create the
|
||||||
|
selection as an 'upside-up' selection, with the anchor at the top, and
|
||||||
|
the next cursor motion will behave in an unexpected way—our second
|
||||||
|
up-arrow press in the example above will not do anything, since it is
|
||||||
|
interpreted in exactly the same way as the first.</p>
|
||||||
|
|
||||||
|
<p>No problem. We'll just, ehm, detect that the selection is
|
||||||
|
upside-down (you can tell by the way it was created), and then, when
|
||||||
|
an upside-down selection is present, and a cursor-moving key is
|
||||||
|
pressed in combination with shift, we quickly collapse the selection
|
||||||
|
in the textarea to its start, allow the key to take effect, and then
|
||||||
|
combine its new head with its old anchor to get the <em>real</em>
|
||||||
|
selection.</p>
|
||||||
|
|
||||||
|
<p>In short, scary hacks could not be avoided entirely in CodeMirror
|
||||||
|
2.</p>
|
||||||
|
|
||||||
|
<p>And, the observant reader might ask, how do you even know that a
|
||||||
|
key combo is a cursor-moving combo, if you claim you support any
|
||||||
|
native key bindings? Well, we don't, but we can learn. The editor
|
||||||
|
keeps a set known cursor-movement combos (initialized to the
|
||||||
|
predictable defaults), and updates this set when it observes that
|
||||||
|
pressing a certain key had (only) the effect of moving the cursor.
|
||||||
|
This, of course, doesn't work if the first time the key is used was
|
||||||
|
for extending an inverted selection, but it works most of the
|
||||||
|
time.</p>
|
||||||
|
|
||||||
|
<h2 id="update">Intelligent Updating</h2>
|
||||||
|
|
||||||
|
<p>One thing that always comes up when you have a complicated internal
|
||||||
|
state that's reflected in some user-visible external representation
|
||||||
|
(in this case, the displayed code and the textarea's content) is
|
||||||
|
keeping the two in sync. The naive way is to just update the display
|
||||||
|
every time you change your state, but this is not only error prone
|
||||||
|
(you'll forget), it also easily leads to duplicate work on big,
|
||||||
|
composite operations. Then you start passing around flags indicating
|
||||||
|
whether the display should be updated in an attempt to be efficient
|
||||||
|
again and, well, at that point you might as well give up completely.</p>
|
||||||
|
|
||||||
|
<p>I did go down that road, but then switched to a much simpler model:
|
||||||
|
simply keep track of all the things that have been changed during an
|
||||||
|
action, and then, only at the end, use this information to update the
|
||||||
|
user-visible display.</p>
|
||||||
|
|
||||||
|
<p>CodeMirror uses a concept of <em>operations</em>, which start by
|
||||||
|
calling a specific set-up function that clears the state and end by
|
||||||
|
calling another function that reads this state and does the required
|
||||||
|
updating. Most event handlers, and all the user-visible methods that
|
||||||
|
change state are wrapped like this. There's a method
|
||||||
|
called <code>operation</code> that accepts a function, and returns
|
||||||
|
another function that wraps the given function as an operation.</p>
|
||||||
|
|
||||||
|
<p>It's trivial to extend this (as CodeMirror does) to detect nesting,
|
||||||
|
and, when an operation is started inside an operation, simply
|
||||||
|
increment the nesting count, and only do the updating when this count
|
||||||
|
reaches zero again.</p>
|
||||||
|
|
||||||
|
<p>If we have a set of changed ranges and know the currently shown
|
||||||
|
range, we can (with some awkward code to deal with the fact that
|
||||||
|
changes can add and remove lines, so we're dealing with a changing
|
||||||
|
coordinate system) construct a map of the ranges that were left
|
||||||
|
intact. We can then compare this map with the part of the document
|
||||||
|
that's currently visible (based on scroll offset and editor height) to
|
||||||
|
determine whether something needs to be updated.</p>
|
||||||
|
|
||||||
|
<p>CodeMirror uses two update algorithms—a full refresh, where it just
|
||||||
|
discards the whole part of the DOM that contains the edited text and
|
||||||
|
rebuilds it, and a patch algorithm, where it uses the information
|
||||||
|
about changed and intact ranges to update only the out-of-date parts
|
||||||
|
of the DOM. When more than 30 percent (which is the current heuristic,
|
||||||
|
might change) of the lines need to be updated, the full refresh is
|
||||||
|
chosen (since it's faster to do than painstakingly finding and
|
||||||
|
updating all the changed lines), in the other case it does the
|
||||||
|
patching (so that, if you scroll a line or select another character,
|
||||||
|
the whole screen doesn't have to be
|
||||||
|
re-rendered). <span class="update">[the full-refresh
|
||||||
|
algorithm was dropped, it wasn't really faster than the patching
|
||||||
|
one]</span></p>
|
||||||
|
|
||||||
|
<p>All updating uses <code>innerHTML</code> rather than direct DOM
|
||||||
|
manipulation, since that still seems to be by far the fastest way to
|
||||||
|
build documents. There's a per-line function that combines the
|
||||||
|
highlighting, <a href="manual.html#markText">marking</a>, and
|
||||||
|
selection info for that line into a snippet of HTML. The patch updater
|
||||||
|
uses this to reset individual lines, the refresh updater builds an
|
||||||
|
HTML chunk for the whole visible document at once, and then uses a
|
||||||
|
single <code>innerHTML</code> update to do the refresh.</p>
|
||||||
|
|
||||||
|
<h2 id="parse">Parsers can be Simple</h2>
|
||||||
|
|
||||||
|
<p>When I wrote CodeMirror 1, I
|
||||||
|
thought <a href="http://codemirror.net/story.html#parser">interruptable
|
||||||
|
parsers</a> were a hugely scary and complicated thing, and I used a
|
||||||
|
bunch of heavyweight abstractions to keep this supposed complexity
|
||||||
|
under control: parsers
|
||||||
|
were <a href="http://bob.pythonmac.org/archives/2005/07/06/iteration-in-javascript/">iterators</a>
|
||||||
|
that consumed input from another iterator, and used funny
|
||||||
|
closure-resetting tricks to copy and resume themselves.</p>
|
||||||
|
|
||||||
|
<p>This made for a rather nice system, in that parsers formed strictly
|
||||||
|
separate modules, and could be composed in predictable ways.
|
||||||
|
Unfortunately, it was quite slow (stacking three or four iterators on
|
||||||
|
top of each other), and extremely intimidating to people not used to a
|
||||||
|
functional programming style.</p>
|
||||||
|
|
||||||
|
<p>With a few small changes, however, we can keep all those
|
||||||
|
advantages, but simplify the API and make the whole thing less
|
||||||
|
indirect and inefficient. CodeMirror
|
||||||
|
2's <a href="manual.html#modeapi">mode API</a> uses explicit state
|
||||||
|
objects, and makes the parser/tokenizer a function that simply takes a
|
||||||
|
state and a character stream abstraction, advances the stream one
|
||||||
|
token, and returns the way the token should be styled. This state may
|
||||||
|
be copied, optionally in a mode-defined way, in order to be able to
|
||||||
|
continue a parse at a given point. Even someone who's never touched a
|
||||||
|
lambda in his life can understand this approach. Additionally, far
|
||||||
|
fewer objects are allocated in the course of parsing now.</p>
|
||||||
|
|
||||||
|
<p>The biggest speedup comes from the fact that the parsing no longer
|
||||||
|
has to touch the DOM though. In CodeMirror 1, on an older browser, you
|
||||||
|
could <em>see</em> the parser work its way through the document,
|
||||||
|
managing some twenty lines in each 50-millisecond time slice it got. It
|
||||||
|
was reading its input from the DOM, and updating the DOM as it went
|
||||||
|
along, which any experienced JavaScript programmer will immediately
|
||||||
|
spot as a recipe for slowness. In CodeMirror 2, the parser usually
|
||||||
|
finishes the whole document in a single 100-millisecond time slice—it
|
||||||
|
manages some 1500 lines during that time on Chrome. All it has to do
|
||||||
|
is munge strings, so there is no real reason for it to be slow
|
||||||
|
anymore.</p>
|
||||||
|
|
||||||
|
<h2 id="summary">What Gives?</h2>
|
||||||
|
|
||||||
|
<p>Given all this, what can you expect from CodeMirror 2?</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
|
||||||
|
<li><strong>Small.</strong> the base library is
|
||||||
|
some <span class="update">45k</span> when minified
|
||||||
|
now, <span class="update">17k</span> when gzipped. It's smaller than
|
||||||
|
its own logo.</li>
|
||||||
|
|
||||||
|
<li><strong>Lightweight.</strong> CodeMirror 2 initializes very
|
||||||
|
quickly, and does almost no work when it is not focused. This means
|
||||||
|
you can treat it almost like a textarea, have multiple instances on a
|
||||||
|
page without trouble.</li>
|
||||||
|
|
||||||
|
<li><strong>Huge document support.</strong> Since highlighting is
|
||||||
|
really fast, and no DOM structure is being built for non-visible
|
||||||
|
content, you don't have to worry about locking up your browser when a
|
||||||
|
user enters a megabyte-sized document.</li>
|
||||||
|
|
||||||
|
<li><strong>Extended API.</strong> Some things kept coming up in the
|
||||||
|
mailing list, such as marking pieces of text or lines, which were
|
||||||
|
extremely hard to do with CodeMirror 1. The new version has proper
|
||||||
|
support for these built in.</li>
|
||||||
|
|
||||||
|
<li><strong>Tab support.</strong> Tabs inside editable documents were,
|
||||||
|
for some reason, a no-go. At least six different people announced they
|
||||||
|
were going to add tab support to CodeMirror 1, none survived (I mean,
|
||||||
|
none delivered a working version). CodeMirror 2 no longer removes tabs
|
||||||
|
from your document.</li>
|
||||||
|
|
||||||
|
<li><strong>Sane styling.</strong> <code>iframe</code> nodes aren't
|
||||||
|
really known for respecting document flow. Now that an editor instance
|
||||||
|
is a plain <code>div</code> element, it is much easier to size it to
|
||||||
|
fit the surrounding elements. You don't even have to make it scroll if
|
||||||
|
you do not <a href="../demo/resize.html">want to</a>.</li>
|
||||||
|
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p>On the downside, a CodeMirror 2 instance is <em>not</em> a native
|
||||||
|
editable component. Though it does its best to emulate such a
|
||||||
|
component as much as possible, there is functionality that browsers
|
||||||
|
just do not allow us to hook into. Doing select-all from the context
|
||||||
|
menu, for example, is not currently detected by CodeMirror.</p>
|
||||||
|
|
||||||
|
<p id="changes" style="margin-top: 2em;"><span style="font-weight:
|
||||||
|
bold">[Updates from November 13th 2011]</span> Recently, I've made
|
||||||
|
some changes to the codebase that cause some of the text above to no
|
||||||
|
longer be current. I've left the text intact, but added markers at the
|
||||||
|
passages that are now inaccurate. The new situation is described
|
||||||
|
below.</p>
|
||||||
|
|
||||||
|
<h2 id="btree">Content Representation</h2>
|
||||||
|
|
||||||
|
<p>The original implementation of CodeMirror 2 represented the
|
||||||
|
document as a flat array of line objects. This worked well—splicing
|
||||||
|
arrays will require the part of the array after the splice to be
|
||||||
|
moved, but this is basically just a simple <code>memmove</code> of a
|
||||||
|
bunch of pointers, so it is cheap even for huge documents.</p>
|
||||||
|
|
||||||
|
<p>However, I recently added line wrapping and code folding (line
|
||||||
|
collapsing, basically). Once lines start taking up a non-constant
|
||||||
|
amount of vertical space, looking up a line by vertical position
|
||||||
|
(which is needed when someone clicks the document, and to determine
|
||||||
|
the visible part of the document during scrolling) can only be done
|
||||||
|
with a linear scan through the whole array, summing up line heights as
|
||||||
|
you go. Seeing how I've been going out of my way to make big documents
|
||||||
|
fast, this is not acceptable.</p>
|
||||||
|
|
||||||
|
<p>The new representation is based on a B-tree. The leaves of the tree
|
||||||
|
contain arrays of line objects, with a fixed minimum and maximum size,
|
||||||
|
and the non-leaf nodes simply hold arrays of child nodes. Each node
|
||||||
|
stores both the amount of lines that live below them and the vertical
|
||||||
|
space taken up by these lines. This allows the tree to be indexed both
|
||||||
|
by line number and by vertical position, and all access has
|
||||||
|
logarithmic complexity in relation to the document size.</p>
|
||||||
|
|
||||||
|
<p>I gave line objects and tree nodes parent pointers, to the node
|
||||||
|
above them. When a line has to update its height, it can simply walk
|
||||||
|
these pointers to the top of the tree, adding or subtracting the
|
||||||
|
difference in height from each node it encounters. The parent pointers
|
||||||
|
also make it cheaper (in complexity terms, the difference is probably
|
||||||
|
tiny in normal-sized documents) to find the current line number when
|
||||||
|
given a line object. In the old approach, the whole document array had
|
||||||
|
to be searched. Now, we can just walk up the tree and count the sizes
|
||||||
|
of the nodes coming before us at each level.</p>
|
||||||
|
|
||||||
|
<p>I chose B-trees, not regular binary trees, mostly because they
|
||||||
|
allow for very fast bulk insertions and deletions. When there is a big
|
||||||
|
change to a document, it typically involves adding, deleting, or
|
||||||
|
replacing a chunk of subsequent lines. In a regular balanced tree, all
|
||||||
|
these inserts or deletes would have to be done separately, which could
|
||||||
|
be really expensive. In a B-tree, to insert a chunk, you just walk
|
||||||
|
down the tree once to find where it should go, insert them all in one
|
||||||
|
shot, and then break up the node if needed. This breaking up might
|
||||||
|
involve breaking up nodes further up, but only requires a single pass
|
||||||
|
back up the tree. For deletion, I'm somewhat lax in keeping things
|
||||||
|
balanced—I just collapse nodes into a leaf when their child count goes
|
||||||
|
below a given number. This means that there are some weird editing
|
||||||
|
patterns that may result in a seriously unbalanced tree, but even such
|
||||||
|
an unbalanced tree will perform well, unless you spend a day making
|
||||||
|
strangely repeating edits to a really big document.</p>
|
||||||
|
|
||||||
|
<h2 id="keymap">Keymaps</h2>
|
||||||
|
|
||||||
|
<p><a href="#approach">Above</a>, I claimed that directly catching key
|
||||||
|
events for things like cursor movement is impractical because it
|
||||||
|
requires some browser-specific kludges. I then proceeded to explain
|
||||||
|
some awful <a href="#selection">hacks</a> that were needed to make it
|
||||||
|
possible for the selection changes to be detected through the
|
||||||
|
textarea. In fact, the second hack is about as bad as the first.</p>
|
||||||
|
|
||||||
|
<p>On top of that, in the presence of user-configurable tab sizes and
|
||||||
|
collapsed and wrapped lines, lining up cursor movement in the textarea
|
||||||
|
with what's visible on the screen becomes a nightmare. Thus, I've
|
||||||
|
decided to move to a model where the textarea's selection is no longer
|
||||||
|
depended on.</p>
|
||||||
|
|
||||||
|
<p>So I moved to a model where all cursor movement is handled by my
|
||||||
|
own code. This adds support for a goal column, proper interaction of
|
||||||
|
cursor movement with collapsed lines, and makes it possible for
|
||||||
|
vertical movement to move through wrapped lines properly, instead of
|
||||||
|
just treating them like non-wrapped lines.</p>
|
||||||
|
|
||||||
|
<p>The key event handlers now translate the key event into a string,
|
||||||
|
something like <code>Ctrl-Home</code> or <code>Shift-Cmd-R</code>, and
|
||||||
|
use that string to look up an action to perform. To make keybinding
|
||||||
|
customizable, this lookup goes through
|
||||||
|
a <a href="manual.html#option_keyMap">table</a>, using a scheme that
|
||||||
|
allows such tables to be chained together (for example, the default
|
||||||
|
Mac bindings fall through to a table named 'emacsy', which defines
|
||||||
|
basic Emacs-style bindings like <code>Ctrl-F</code>, and which is also
|
||||||
|
used by the custom Emacs bindings).</p>
|
||||||
|
|
||||||
|
<p>A new
|
||||||
|
option <a href="manual.html#option_extraKeys"><code>extraKeys</code></a>
|
||||||
|
allows ad-hoc keybindings to be defined in a much nicer way than what
|
||||||
|
was possible with the
|
||||||
|
old <a href="manual.html#option_onKeyEvent"><code>onKeyEvent</code></a>
|
||||||
|
callback. You simply provide an object mapping key identifiers to
|
||||||
|
functions, instead of painstakingly looking at raw key events.</p>
|
||||||
|
|
||||||
|
<p>Built-in commands map to strings, rather than functions, for
|
||||||
|
example <code>"goLineUp"</code> is the default action bound to the up
|
||||||
|
arrow key. This allows new keymaps to refer to them without
|
||||||
|
duplicating any code. New commands can be defined by assigning to
|
||||||
|
the <code>CodeMirror.commands</code> object, which maps such commands
|
||||||
|
to functions.</p>
|
||||||
|
|
||||||
|
<p>The hidden textarea now only holds the current selection, with no
|
||||||
|
extra characters around it. This has a nice advantage: polling for
|
||||||
|
input becomes much, much faster. If there's a big selection, this text
|
||||||
|
does not have to be read from the textarea every time—when we poll,
|
||||||
|
just noticing that something is still selected is enough to tell us
|
||||||
|
that no new text was typed.</p>
|
||||||
|
|
||||||
|
<p>The reason that cheap polling is important is that many browsers do
|
||||||
|
not fire useful events on IME (input method engine) input, which is
|
||||||
|
the thing where people inputting a language like Japanese or Chinese
|
||||||
|
use multiple keystrokes to create a character or sequence of
|
||||||
|
characters. Most modern browsers fire <code>input</code> when the
|
||||||
|
composing is finished, but many don't fire anything when the character
|
||||||
|
is updated <em>during</em> composition. So we poll, whenever the
|
||||||
|
editor is focused, to provide immediate updates of the display.</p>
|
||||||
|
|
||||||
|
</div><div class="rightsmall blk">
|
||||||
|
|
||||||
|
<h2>Contents</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="#intro">Introduction</a></li>
|
||||||
|
<li><a href="#approach">General Approach</a></li>
|
||||||
|
<li><a href="#input">Input</a></li>
|
||||||
|
<li><a href="#selection">Selection</a></li>
|
||||||
|
<li><a href="#update">Intelligent Updating</a></li>
|
||||||
|
<li><a href="#parse">Parsing</a></li>
|
||||||
|
<li><a href="#summary">What Gives?</a></li>
|
||||||
|
<li><a href="#btree">Content Representation</a></li>
|
||||||
|
<li><a href="#keymap">Key Maps</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
</div></div>
|
||||||
|
|
||||||
|
<div style="height: 2em"> </div>
|
||||||
|
|
||||||
|
</body></html>
|
1242
codemirror-2.36/doc/manual.html
Normal file
1242
codemirror-2.36/doc/manual.html
Normal file
File diff suppressed because it is too large
Load Diff
356
codemirror-2.36/doc/oldrelease.html
Normal file
356
codemirror-2.36/doc/oldrelease.html
Normal file
|
@ -0,0 +1,356 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
<link rel="alternate" href="http://twitter.com/statuses/user_timeline/242283288.rss" type="application/rss+xml"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* Old release
|
||||||
|
history */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="rel">27-02-2012: <a href="http://codemirror.net/codemirror-2.22.zip">Version 2.22</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Allow <a href="manual.html#keymaps">key handlers</a> to pass up events, allow binding characters.</li>
|
||||||
|
<li>Add <a href="manual.html#option_autoClearEmptyLines"><code>autoClearEmptyLines</code></a> option.</li>
|
||||||
|
<li>Properly use tab stops when rendering tabs.</li>
|
||||||
|
<li>Make PHP mode more robust.</li>
|
||||||
|
<li>Support indentation blocks in <a href="manual.html#util_foldcode">code folder</a>.</li>
|
||||||
|
<li>Add a script for <a href="manual.html#util_match-highlighter">highlighting instances of the selection</a>.</li>
|
||||||
|
<li>New <a href="../mode/properties/index.html">.properties</a> mode.</li>
|
||||||
|
<li>Fix many bugs.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">27-01-2012: <a href="http://codemirror.net/codemirror-2.21.zip">Version 2.21</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Added <a href="../mode/less/index.html">LESS</a>, <a href="../mode/mysql/index.html">MySQL</a>,
|
||||||
|
<a href="../mode/go/index.html">Go</a>, and <a href="../mode/verilog/index.html">Verilog</a> modes.</li>
|
||||||
|
<li>Add <a href="manual.html#option_smartIndent"><code>smartIndent</code></a>
|
||||||
|
option.</li>
|
||||||
|
<li>Support a cursor in <a href="manual.html#option_readOnly"><code>readOnly</code></a>-mode.</li>
|
||||||
|
<li>Support assigning multiple styles to a token.</li>
|
||||||
|
<li>Use a new approach to drawing the selection.</li>
|
||||||
|
<li>Add <a href="manual.html#scrollTo"><code>scrollTo</code></a> method.</li>
|
||||||
|
<li>Allow undo/redo events to span non-adjacent lines.</li>
|
||||||
|
<li>Lots and lots of bugfixes.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">20-12-2011: <a href="http://codemirror.net/codemirror-2.2.zip">Version 2.2</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Slightly incompatible API changes. Read <a href="upgrade_v2.2.html">this</a>.</li>
|
||||||
|
<li>New approach
|
||||||
|
to <a href="manual.html#option_extraKeys">binding</a> keys,
|
||||||
|
support for <a href="manual.html#option_keyMap">custom
|
||||||
|
bindings</a>.</li>
|
||||||
|
<li>Support for overwrite (insert).</li>
|
||||||
|
<li><a href="manual.html#option_tabSize">Custom-width</a>
|
||||||
|
and <a href="../demo/visibletabs.html">stylable</a> tabs.</li>
|
||||||
|
<li>Moved more code into <a href="manual.html#addons">add-on scripts</a>.</li>
|
||||||
|
<li>Support for sane vertical cursor movement in wrapped lines.</li>
|
||||||
|
<li>More reliable handling of
|
||||||
|
editing <a href="manual.html#markText">marked text</a>.</li>
|
||||||
|
<li>Add minimal <a href="../demo/emacs.html">emacs</a>
|
||||||
|
and <a href="../demo/vim.html">vim</a> bindings.</li>
|
||||||
|
<li>Rename <code>coordsFromIndex</code>
|
||||||
|
to <a href="manual.html#posFromIndex"><code>posFromIndex</code></a>,
|
||||||
|
add <a href="manual.html#indexFromPos"><code>indexFromPos</code></a>
|
||||||
|
method.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">21-11-2011: <a href="http://codemirror.net/codemirror-2.18.zip">Version 2.18</a>:</p>
|
||||||
|
<p class="rel-note">Fixes <code>TextMarker.clear</code>, which is broken in 2.17.</p>
|
||||||
|
|
||||||
|
<p class="rel">21-11-2011: <a href="http://codemirror.net/codemirror-2.17.zip">Version 2.17</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add support for <a href="manual.html#option_lineWrapping">line
|
||||||
|
wrapping</a> and <a href="manual.html#hideLine">code
|
||||||
|
folding</a>.</li>
|
||||||
|
<li>Add <a href="../mode/gfm/index.html">Github-style Markdown</a> mode.</li>
|
||||||
|
<li>Add <a href="../theme/monokai.css">Monokai</a>
|
||||||
|
and <a href="../theme/rubyblue.css">Rubyblue</a> themes.</li>
|
||||||
|
<li>Add <a href="manual.html#setBookmark"><code>setBookmark</code></a> method.</li>
|
||||||
|
<li>Move some of the demo code into reusable components
|
||||||
|
under <a href="../lib/util/"><code>lib/util</code></a>.</li>
|
||||||
|
<li>Make screen-coord-finding code faster and more reliable.</li>
|
||||||
|
<li>Fix drag-and-drop in Firefox.</li>
|
||||||
|
<li>Improve support for IME.</li>
|
||||||
|
<li>Speed up content rendering.</li>
|
||||||
|
<li>Fix browser's built-in search in Webkit.</li>
|
||||||
|
<li>Make double- and triple-click work in IE.</li>
|
||||||
|
<li>Various fixes to modes.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">27-10-2011: <a href="http://codemirror.net/codemirror-2.16.zip">Version 2.16</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add <a href="../mode/perl/index.html">Perl</a>, <a href="../mode/rust/index.html">Rust</a>, <a href="../mode/tiddlywiki/index.html">TiddlyWiki</a>, and <a href="../mode/groovy/index.html">Groovy</a> modes.</li>
|
||||||
|
<li>Dragging text inside the editor now moves, rather than copies.</li>
|
||||||
|
<li>Add a <a href="manual.html#coordsFromIndex"><code>coordsFromIndex</code></a> method.</li>
|
||||||
|
<li><strong>API change</strong>: <code>setValue</code> now no longer clears history. Use <a href="manual.html#clearHistory"><code>clearHistory</code></a> for that.</li>
|
||||||
|
<li><strong>API change</strong>: <a href="manual.html#markText"><code>markText</code></a> now
|
||||||
|
returns an object with <code>clear</code> and <code>find</code>
|
||||||
|
methods. Marked text is now more robust when edited.</li>
|
||||||
|
<li>Fix editing code with tabs in Internet Explorer.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">26-09-2011: <a href="http://codemirror.net/codemirror-2.15.zip">Version 2.15</a>:</p>
|
||||||
|
<p class="rel-note">Fix bug that snuck into 2.14: Clicking the
|
||||||
|
character that currently has the cursor didn't re-focus the
|
||||||
|
editor.</p>
|
||||||
|
|
||||||
|
<p class="rel">26-09-2011: <a href="http://codemirror.net/codemirror-2.14.zip">Version 2.14</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add <a href="../mode/clojure/index.html">Clojure</a>, <a href="../mode/pascal/index.html">Pascal</a>, <a href="../mode/ntriples/index.html">NTriples</a>, <a href="../mode/jinja2/index.html">Jinja2</a>, and <a href="../mode/markdown/index.html">Markdown</a> modes.</li>
|
||||||
|
<li>Add <a href="../theme/cobalt.css">Cobalt</a> and <a href="../theme/eclipse.css">Eclipse</a> themes.</li>
|
||||||
|
<li>Add a <a href="manual.html#option_fixedGutter"><code>fixedGutter</code></a> option.</li>
|
||||||
|
<li>Fix bug with <code>setValue</code> breaking cursor movement.</li>
|
||||||
|
<li>Make gutter updates much more efficient.</li>
|
||||||
|
<li>Allow dragging of text out of the editor (on modern browsers).</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
|
||||||
|
<p class="rel">23-08-2011: <a href="http://codemirror.net/codemirror-2.13.zip">Version 2.13</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add <a href="../mode/ruby/index.html">Ruby</a>, <a href="../mode/r/index.html">R</a>, <a href="../mode/coffeescript/index.html">CoffeeScript</a>, and <a href="../mode/velocity/index.html">Velocity</a> modes.</li>
|
||||||
|
<li>Add <a href="manual.html#getGutterElement"><code>getGutterElement</code></a> to API.</li>
|
||||||
|
<li>Several fixes to scrolling and positioning.</li>
|
||||||
|
<li>Add <a href="manual.html#option_smartHome"><code>smartHome</code></a> option.</li>
|
||||||
|
<li>Add an experimental <a href="../mode/xmlpure/index.html">pure XML</a> mode.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">25-07-2011: <a href="http://codemirror.net/codemirror-2.12.zip">Version 2.12</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add a <a href="../mode/sparql/index.html">SPARQL</a> mode.</li>
|
||||||
|
<li>Fix bug with cursor jumping around in an unfocused editor in IE.</li>
|
||||||
|
<li>Allow key and mouse events to bubble out of the editor. Ignore widget clicks.</li>
|
||||||
|
<li>Solve cursor flakiness after undo/redo.</li>
|
||||||
|
<li>Fix block-reindent ignoring the last few lines.</li>
|
||||||
|
<li>Fix parsing of multi-line attrs in XML mode.</li>
|
||||||
|
<li>Use <code>innerHTML</code> for HTML-escaping.</li>
|
||||||
|
<li>Some fixes to indentation in C-like mode.</li>
|
||||||
|
<li>Shrink horiz scrollbars when long lines removed.</li>
|
||||||
|
<li>Fix width feedback loop bug that caused the width of an inner DIV to shrink.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">04-07-2011: <a href="http://codemirror.net/codemirror-2.11.zip">Version 2.11</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add a <a href="../mode/scheme/index.html">Scheme mode</a>.</li>
|
||||||
|
<li>Add a <code>replace</code> method to search cursors, for cursor-preserving replacements.</li>
|
||||||
|
<li>Make the <a href="../mode/clike/index.html">C-like mode</a> mode more customizable.</li>
|
||||||
|
<li>Update XML mode to spot mismatched tags.</li>
|
||||||
|
<li>Add <code>getStateAfter</code> API and <code>compareState</code> mode API methods for finer-grained mode magic.</li>
|
||||||
|
<li>Add a <code>getScrollerElement</code> API method to manipulate the scrolling DIV.</li>
|
||||||
|
<li>Fix drag-and-drop for Firefox.</li>
|
||||||
|
<li>Add a C# configuration for the <a href="../mode/clike/index.html">C-like mode</a>.</li>
|
||||||
|
<li>Add <a href="../demo/fullscreen.html">full-screen editing</a> and <a href="../demo/changemode.html">mode-changing</a> demos.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">07-06-2011: <a href="http://codemirror.net/codemirror-2.1.zip">Version 2.1</a>:</p>
|
||||||
|
<p class="rel-note">Add
|
||||||
|
a <a href="manual.html#option_theme">theme</a> system
|
||||||
|
(<a href="../demo/theme.html">demo</a>). Note that this is not
|
||||||
|
backwards-compatible—you'll have to update your styles and
|
||||||
|
modes!</p>
|
||||||
|
|
||||||
|
<p class="rel">07-06-2011: <a href="http://codemirror.net/codemirror-2.02.zip">Version 2.02</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add a <a href="../mode/lua/index.html">Lua mode</a>.</li>
|
||||||
|
<li>Fix reverse-searching for a regexp.</li>
|
||||||
|
<li>Empty lines can no longer break highlighting.</li>
|
||||||
|
<li>Rework scrolling model (the outer wrapper no longer does the scrolling).</li>
|
||||||
|
<li>Solve horizontal jittering on long lines.</li>
|
||||||
|
<li>Add <a href="../demo/runmode.html">runmode.js</a>.</li>
|
||||||
|
<li>Immediately re-highlight text when typing.</li>
|
||||||
|
<li>Fix problem with 'sticking' horizontal scrollbar.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">26-05-2011: <a href="http://codemirror.net/codemirror-2.01.zip">Version 2.01</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add a <a href="../mode/smalltalk/index.html">Smalltalk mode</a>.</li>
|
||||||
|
<li>Add a <a href="../mode/rst/index.html">reStructuredText mode</a>.</li>
|
||||||
|
<li>Add a <a href="../mode/python/index.html">Python mode</a>.</li>
|
||||||
|
<li>Add a <a href="../mode/plsql/index.html">PL/SQL mode</a>.</li>
|
||||||
|
<li><code>coordsChar</code> now works</li>
|
||||||
|
<li>Fix a problem where <code>onCursorActivity</code> interfered with <code>onChange</code>.</li>
|
||||||
|
<li>Fix a number of scrolling and mouse-click-position glitches.</li>
|
||||||
|
<li>Pass information about the changed lines to <code>onChange</code>.</li>
|
||||||
|
<li>Support cmd-up/down on OS X.</li>
|
||||||
|
<li>Add triple-click line selection.</li>
|
||||||
|
<li>Don't handle shift when changing the selection through the API.</li>
|
||||||
|
<li>Support <code>"nocursor"</code> mode for <code>readOnly</code> option.</li>
|
||||||
|
<li>Add an <code>onHighlightComplete</code> option.</li>
|
||||||
|
<li>Fix the context menu for Firefox.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">28-03-2011: <a href="http://codemirror.net/codemirror-2.0.zip">Version 2.0</a>:</p>
|
||||||
|
<p class="rel-note">CodeMirror 2 is a complete rewrite that's
|
||||||
|
faster, smaller, simpler to use, and less dependent on browser
|
||||||
|
quirks. See <a href="internals.html">this</a>
|
||||||
|
and <a href="http://groups.google.com/group/codemirror/browse_thread/thread/5a8e894024a9f580">this</a>
|
||||||
|
for more information.</a>
|
||||||
|
|
||||||
|
<p class="rel">28-03-2011: <a href="http://codemirror.net/codemirror-1.0.zip">Version 1.0</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Fix error when debug history overflows.</li>
|
||||||
|
<li>Refine handling of C# verbatim strings.</li>
|
||||||
|
<li>Fix some issues with JavaScript indentation.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">22-02-2011: <a href="https://github.com/marijnh/codemirror/tree/beta2">Version 2.0 beta 2</a>:</p>
|
||||||
|
<p class="rel-note">Somewhat more mature API, lots of bugs shaken out.</a>
|
||||||
|
|
||||||
|
<p class="rel">17-02-2011: <a href="http://codemirror.net/codemirror-0.94.zip">Version 0.94</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li><code>tabMode: "spaces"</code> was modified slightly (now indents when something is selected).</li>
|
||||||
|
<li>Fixes a bug that would cause the selection code to break on some IE versions.</li>
|
||||||
|
<li>Disabling spell-check on WebKit browsers now works.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">08-02-2011: <a href="http://codemirror.net/">Version 2.0 beta 1</a>:</p>
|
||||||
|
<p class="rel-note">CodeMirror 2 is a complete rewrite of
|
||||||
|
CodeMirror, no longer depending on an editable frame.</p>
|
||||||
|
|
||||||
|
<p class="rel">19-01-2011: <a href="http://codemirror.net/codemirror-0.93.zip">Version 0.93</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Added a <a href="contrib/regex/index.html">Regular Expression</a> parser.</li>
|
||||||
|
<li>Fixes to the PHP parser.</li>
|
||||||
|
<li>Support for regular expression in search/replace.</li>
|
||||||
|
<li>Add <code>save</code> method to instances created with <code>fromTextArea</code>.</li>
|
||||||
|
<li>Add support for MS T-SQL in the SQL parser.</li>
|
||||||
|
<li>Support use of CSS classes for highlighting brackets.</li>
|
||||||
|
<li>Fix yet another hang with line-numbering in hidden editors.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">17-12-2010: <a href="http://codemirror.net/codemirror-0.92.zip">Version 0.92</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Make CodeMirror work in XHTML documents.</li>
|
||||||
|
<li>Fix bug in handling of backslashes in Python strings.</li>
|
||||||
|
<li>The <code>styleNumbers</code> option is now officially
|
||||||
|
supported and documented.</li>
|
||||||
|
<li><code>onLineNumberClick</code> option added.</li>
|
||||||
|
<li>More consistent names <code>onLoad</code> and
|
||||||
|
<code>onCursorActivity</code> callbacks. Old names still work, but
|
||||||
|
are deprecated.</li>
|
||||||
|
<li>Add a <a href="contrib/freemarker/index.html">Freemarker</a> mode.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">11-11-2010: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.91.zip">Version 0.91</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Adds support for <a href="contrib/java">Java</a>.</li>
|
||||||
|
<li>Small additions to the <a href="contrib/php">PHP</a> and <a href="contrib/sql">SQL</a> parsers.</li>
|
||||||
|
<li>Work around various <a href="https://bugs.webkit.org/show_bug.cgi?id=47806">Webkit</a> <a href="https://bugs.webkit.org/show_bug.cgi?id=23474">issues</a>.</li>
|
||||||
|
<li>Fix <code>toTextArea</code> to update the code in the textarea.</li>
|
||||||
|
<li>Add a <code>noScriptCaching</code> option (hack to ease development).</li>
|
||||||
|
<li>Make sub-modes of <a href="mixedtest.html">HTML mixed</a> mode configurable.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">02-10-2010: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.9.zip">Version 0.9</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add support for searching backwards.</li>
|
||||||
|
<li>There are now parsers for <a href="contrib/scheme/index.html">Scheme</a>, <a href="contrib/xquery/index.html">XQuery</a>, and <a href="contrib/ometa/index.html">OmetaJS</a>.</li>
|
||||||
|
<li>Makes <code>height: "dynamic"</code> more robust.</li>
|
||||||
|
<li>Fixes bug where paste did not work on OS X.</li>
|
||||||
|
<li>Add a <code>enterMode</code> and <code>electricChars</code> options to make indentation even more customizable.</li>
|
||||||
|
<li>Add <code>firstLineNumber</code> option.</li>
|
||||||
|
<li>Fix bad handling of <code>@media</code> rules by the CSS parser.</li>
|
||||||
|
<li>Take a new, more robust approach to working around the invisible-last-line bug in WebKit.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">22-07-2010: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.8.zip">Version 0.8</a>:</p>
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Add a <code>cursorCoords</code> method to find the screen
|
||||||
|
coordinates of the cursor.</li>
|
||||||
|
<li>A number of fixes and support for more syntax in the PHP parser.</li>
|
||||||
|
<li>Fix indentation problem with JSON-mode JS parser in Webkit.</li>
|
||||||
|
<li>Add a <a href="compress.html">minification</a> UI.</li>
|
||||||
|
<li>Support a <code>height: dynamic</code> mode, where the editor's
|
||||||
|
height will adjust to the size of its content.</li>
|
||||||
|
<li>Better support for IME input mode.</li>
|
||||||
|
<li>Fix JavaScript parser getting confused when seeing a no-argument
|
||||||
|
function call.</li>
|
||||||
|
<li>Have CSS parser see the difference between selectors and other
|
||||||
|
identifiers.</li>
|
||||||
|
<li>Fix scrolling bug when pasting in a horizontally-scrolled
|
||||||
|
editor.</li>
|
||||||
|
<li>Support <code>toTextArea</code> method in instances created with
|
||||||
|
<code>fromTextArea</code>.</li>
|
||||||
|
<li>Work around new Opera cursor bug that causes the cursor to jump
|
||||||
|
when pressing backspace at the end of a line.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">27-04-2010: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.67.zip">Version
|
||||||
|
0.67</a>:</p>
|
||||||
|
<p class="rel-note">More consistent page-up/page-down behaviour
|
||||||
|
across browsers. Fix some issues with hidden editors looping forever
|
||||||
|
when line-numbers were enabled. Make PHP parser parse
|
||||||
|
<code>"\\"</code> correctly. Have <code>jumpToLine</code> work on
|
||||||
|
line handles, and add <code>cursorLine</code> function to fetch the
|
||||||
|
line handle where the cursor currently is. Add new
|
||||||
|
<code>setStylesheet</code> function to switch style-sheets in a
|
||||||
|
running editor.</p>
|
||||||
|
|
||||||
|
<p class="rel">01-03-2010: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.66.zip">Version
|
||||||
|
0.66</a>:</p>
|
||||||
|
<p class="rel-note">Adds <code>removeLine</code> method to API.
|
||||||
|
Introduces the <a href="contrib/plsql/index.html">PLSQL parser</a>.
|
||||||
|
Marks XML errors by adding (rather than replacing) a CSS class, so
|
||||||
|
that they can be disabled by modifying their style. Fixes several
|
||||||
|
selection bugs, and a number of small glitches.</p>
|
||||||
|
|
||||||
|
<p class="rel">12-11-2009: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.65.zip">Version
|
||||||
|
0.65</a>:</p>
|
||||||
|
<p class="rel-note">Add support for having both line-wrapping and
|
||||||
|
line-numbers turned on, make paren-highlighting style customisable
|
||||||
|
(<code>markParen</code> and <code>unmarkParen</code> config
|
||||||
|
options), work around a selection bug that Opera
|
||||||
|
<em>re</em>introduced in version 10.</p>
|
||||||
|
|
||||||
|
<p class="rel">23-10-2009: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.64.zip">Version
|
||||||
|
0.64</a>:</p>
|
||||||
|
<p class="rel-note">Solves some issues introduced by the
|
||||||
|
paste-handling changes from the previous release. Adds
|
||||||
|
<code>setSpellcheck</code>, <code>setTextWrapping</code>,
|
||||||
|
<code>setIndentUnit</code>, <code>setUndoDepth</code>,
|
||||||
|
<code>setTabMode</code>, and <code>setLineNumbers</code> to
|
||||||
|
customise a running editor. Introduces an <a
|
||||||
|
href="contrib/sql/index.html">SQL</a> parser. Fixes a few small
|
||||||
|
problems in the <a href="contrib/python/index.html">Python</a>
|
||||||
|
parser. And, as usual, add workarounds for various newly discovered
|
||||||
|
browser incompatibilities.</p>
|
||||||
|
|
||||||
|
<p class="rel"><em>31-08-2009</em>: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.63.zip">Version
|
||||||
|
0.63</a>:</p>
|
||||||
|
<p class="rel-note"> Overhaul of paste-handling (less fragile), fixes for several
|
||||||
|
serious IE8 issues (cursor jumping, end-of-document bugs) and a number
|
||||||
|
of small problems.</p>
|
||||||
|
|
||||||
|
<p class="rel"><em>30-05-2009</em>: <a
|
||||||
|
href="http://codemirror.net/codemirror-0.62.zip">Version
|
||||||
|
0.62</a>:</p>
|
||||||
|
<p class="rel-note">Introduces <a href="contrib/python/index.html">Python</a>
|
||||||
|
and <a href="contrib/lua/index.html">Lua</a> parsers. Add
|
||||||
|
<code>setParser</code> (on-the-fly mode changing) and
|
||||||
|
<code>clearHistory</code> methods. Make parsing passes time-based
|
||||||
|
instead of lines-based (see the <code>passTime</code> option).</p>
|
||||||
|
|
||||||
|
</body></html>
|
78
codemirror-2.36/doc/realworld.html
Normal file
78
codemirror-2.36/doc/realworld.html
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror: Real-world uses</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* Real world uses,
|
||||||
|
full list */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p><a href="mailto:marijnh@gmail.com">Contact me</a> if you'd like
|
||||||
|
your project to be added to this list.</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="http://brackets.io">Adobe Brackets</a> (code editor)</li>
|
||||||
|
<li><a href="http://bluegriffon.org/">BlueGriffon</a> (HTML editor)</li>
|
||||||
|
<li><a href="http://cargocollective.com/">Cargo Collective</a> (creative publishing platform)</li>
|
||||||
|
<li><a href="http://www.codebugapp.com/">Codebug</a> (PHP Xdebug front-end)</li>
|
||||||
|
<li><a href="http://code.google.com/p/codemirror2-gwt/">CodeMirror2-GWT</a> (Google Web Toolkit wrapper)</li>
|
||||||
|
<li><a href="http://codepen.io">Codepen</a> (gallery of animations)</li>
|
||||||
|
<li><a href="http://codev.it/">Codev</a> (collaborative IDE)</li>
|
||||||
|
<li><a href="http://ot.substance.io/demo/">Collaborative CodeMirror demo</a> (CodeMirror + operational transforms)</li>
|
||||||
|
<li><a href="http://www.ckwnc.com/">CKWNC</a> (UML editor)</li>
|
||||||
|
<li><a href="http://cssdeck.com/">CSSDeck</a> (CSS showcase)</li>
|
||||||
|
<li><a href="http://ireneros.com/deck/deck.js-codemirror/introduction/#textarea-code">Deck.js integration</a> (slides with editors)</li>
|
||||||
|
<li><a href="http://www.dbninja.com">DbNinja</a> (MySQL access interface)</li>
|
||||||
|
<li><a href="http://elm-lang.org/Examples.elm">Elm language examples</a></li>
|
||||||
|
<li><a href="http://eloquentjavascript.net/chapter1.html">Eloquent JavaScript</a> (book)</li>
|
||||||
|
<li><a href="http://www.fastfig.com/">Fastfig</a> (online computation/math tool)</li>
|
||||||
|
<li><a href="https://metacpan.org/module/Farabi">Farabi</a> (Perl editor)</li>
|
||||||
|
<li><a href="http://blog.pamelafox.org/2012/02/interactive-html5-slides-with-fathomjs.html">FathomJS integration</a> (slides with editors, again)</li>
|
||||||
|
<li><a href="http://tour.golang.org">Go language tour</a></li>
|
||||||
|
<li><a href="https://github.com/github/android">GitHub's Android app</a></li>
|
||||||
|
<li><a href="https://script.google.com/">Google Apps Script</a></li>
|
||||||
|
<li><a href="http://try.haxe.org">Haxe</a> (Haxe Playground) </li>
|
||||||
|
<li><a href="http://megafonweblab.github.com/histone-javascript/">Histone template engine playground</a></li>
|
||||||
|
<li><a href="http://icecoder.net">ICEcoder</a> (web IDE)</li>
|
||||||
|
<li><a href="http://extensions.joomla.org/extensions/edition/editors/8723">Joomla plugin</a></li>
|
||||||
|
<li><a href="http://jsbin.com">jsbin.com</a> (JS playground)</li>
|
||||||
|
<li><a href="http://www.jshint.com/">JSHint</a> (JS linter)</li>
|
||||||
|
<li><a href="http://jumpseller.com/">Jumpseller</a> (online store builder)</li>
|
||||||
|
<li><a href="http://kl1p.com/cmtest/1">kl1p</a> (paste service)</li>
|
||||||
|
<li><a href="http://www.chris-granger.com/2012/04/12/light-table---a-new-ide-concept/">Light Table</a> (experimental IDE)</li>
|
||||||
|
<li><a href="http://www.mergely.com/">Mergely</a> (interactive diffing)</li>
|
||||||
|
<li><a href="https://notex.ch">NoTex</a> (rST authoring)</li>
|
||||||
|
<li><a href="http://clrhome.org/asm/">ORG</a> (z80 assembly IDE)</li>
|
||||||
|
<li><a href="https://github.com/mamacdon/orion-codemirror">Orion-CodeMirror integration</a> (running CodeMirror modes in Orion)</li>
|
||||||
|
<li><a href="http://paperjs.org/">Paper.js</a> (graphics scripting)</li>
|
||||||
|
<li><a href="http://prose.io/">Prose.io</a> (github content editor)</li>
|
||||||
|
<li><a href="http://ql.io/">ql.io</a> (http API query helper)</li>
|
||||||
|
<li><a href="http://qyapp.com">QiYun web app platform</a></li>
|
||||||
|
<li><a href="http://ariya.ofilabs.com/2011/09/hybrid-webnative-desktop-codemirror.html">Qt+Webkit integration</a> (building a desktop CodeMirror app)</li>
|
||||||
|
<li><a href="http://www.sketchpatch.net/labs/livecodelabIntro.html">sketchPatch Livecodelab</a></li>
|
||||||
|
<li><a href="http://www.skulpt.org/">Skulpt</a> (in-browser Python environment)</li>
|
||||||
|
<li><a href="http://www.solidshops.com/">SolidShops</a> (hosted e-commerce platform)</li>
|
||||||
|
<li><a href="http://sqlfiddle.com">SQLFiddle</a> (SQL playground)</li>
|
||||||
|
<li><a href="https://thefiletree.com">The File Tree</a> (collab editor)</li>
|
||||||
|
<li><a href="http://enjalot.com/tributary/2636296/sinwaves.js">Tributary</a> (augmented editing)</li>
|
||||||
|
<li><a href="http://turbopy.com/">TurboPY</a> (web publishing framework)</li>
|
||||||
|
<li><a href="http://webglplayground.net/">WebGL playground</a></li>
|
||||||
|
<li><a href="http://www.wescheme.org/">WeScheme</a> (learning tool)</li>
|
||||||
|
<li><a href="http://wordpress.org/extend/plugins/codemirror-for-codeeditor/">WordPress plugin</a></li>
|
||||||
|
<li><a href="http://www.xosystem.org/home/applications_websites/xosystem_website/xoside_EN.php">XOSide</a> (online editor)</li>
|
||||||
|
<li><a href="http://media.chikuyonok.ru/codemirror2/">Zen Coding</a> (fast XML editing)</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
60
codemirror-2.36/doc/reporting.html
Normal file
60
codemirror-2.36/doc/reporting.html
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror: Reporting Bugs</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
<style>li { margin-top: 1em; }</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* Reporting bugs
|
||||||
|
effectively */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="left">
|
||||||
|
|
||||||
|
<p>So you found a problem in CodeMirror. By all means, report it! Bug
|
||||||
|
reports from users are the main drive behind improvements to
|
||||||
|
CodeMirror. But first, please read over these points:</p>
|
||||||
|
|
||||||
|
<ol>
|
||||||
|
<li>CodeMirror is maintained by volunteers. They don't owe you
|
||||||
|
anything, so be polite. Reports with an indignant or belligerent
|
||||||
|
tone tend to be moved to the bottom of the pile.</li>
|
||||||
|
|
||||||
|
<li>Include information about <strong>the browser in which the
|
||||||
|
problem occurred</strong>. Even if you tested several browsers, and
|
||||||
|
the problem occurred in all of them, mention this fact in the bug
|
||||||
|
report. Also include browser version numbers and the operating
|
||||||
|
system that you're on.</li>
|
||||||
|
|
||||||
|
<li>Mention which release of CodeMirror you're using. Preferably,
|
||||||
|
try also with the current development snapshot, to ensure the
|
||||||
|
problem has not already been fixed.</li>
|
||||||
|
|
||||||
|
<li>Mention very precisely what went wrong. "X is broken" is not a
|
||||||
|
good bug report. What did you expect to happen? What happened
|
||||||
|
instead? Describe the exact steps a maintainer has to take to make
|
||||||
|
the problem occur. We can not fix something that we can not
|
||||||
|
observe.</li>
|
||||||
|
|
||||||
|
<li>If the problem can not be reproduced in any of the demos
|
||||||
|
included in the CodeMirror distribution, please provide an HTML
|
||||||
|
document that demonstrates the problem. The best way to do this is
|
||||||
|
to go to <a href="http://jsbin.com/ihunin/edit">jsbin.com</a>, enter
|
||||||
|
it there, press save, and include the resulting link in your bug
|
||||||
|
report.</li>
|
||||||
|
</ol>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
98
codemirror-2.36/doc/upgrade_v2.2.html
Normal file
98
codemirror-2.36/doc/upgrade_v2.2.html
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror: Upgrading to v2.2</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="docs.css"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* Upgrading to
|
||||||
|
v2.2 */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="left">
|
||||||
|
|
||||||
|
<p>There are a few things in the 2.2 release that require some care
|
||||||
|
when upgrading.</p>
|
||||||
|
|
||||||
|
<h2>No more default.css</h2>
|
||||||
|
|
||||||
|
<p>The default theme is now included
|
||||||
|
in <a href="../lib/codemirror.css"><code>codemirror.css</code></a>, so
|
||||||
|
you do not have to included it separately anymore. (It was tiny, so
|
||||||
|
even if you're not using it, the extra data overhead is negligible.)
|
||||||
|
|
||||||
|
<h2>Different key customization</h2>
|
||||||
|
|
||||||
|
<p>CodeMirror has moved to a system
|
||||||
|
where <a href="manual.html#option_keyMap">keymaps</a> are used to
|
||||||
|
bind behavior to keys. This means <a href="../demo/emacs.html">custom
|
||||||
|
bindings</a> are now possible.</p>
|
||||||
|
|
||||||
|
<p>Three options that influenced key
|
||||||
|
behavior, <code>tabMode</code>, <code>enterMode</code>,
|
||||||
|
and <code>smartHome</code>, are no longer supported. Instead, you can
|
||||||
|
provide custom bindings to influence the way these keys act. This is
|
||||||
|
done through the
|
||||||
|
new <a href="manual.html#option_extraKeys"><code>extraKeys</code></a>
|
||||||
|
option, which can hold an object mapping key names to functionality. A
|
||||||
|
simple example would be:</p>
|
||||||
|
|
||||||
|
<pre> extraKeys: {
|
||||||
|
"Ctrl-S": function(instance) { saveText(instance.getValue()); },
|
||||||
|
"Ctrl-/": "undo"
|
||||||
|
}</pre>
|
||||||
|
|
||||||
|
<p>Keys can be mapped either to functions, which will be given the
|
||||||
|
editor instance as argument, or to strings, which are mapped through
|
||||||
|
functions through the <code>CodeMirror.commands</code> table, which
|
||||||
|
contains all the built-in editing commands, and can be inspected and
|
||||||
|
extended by external code.</p>
|
||||||
|
|
||||||
|
<p>By default, the <code>Home</code> key is bound to
|
||||||
|
the <code>"goLineStartSmart"</code> command, which moves the cursor to
|
||||||
|
the first non-whitespace character on the line. You can set do this to
|
||||||
|
make it always go to the very start instead:</p>
|
||||||
|
|
||||||
|
<pre> extraKeys: {"Home": "goLineStart"}</pre>
|
||||||
|
|
||||||
|
<p>Similarly, <code>Enter</code> is bound
|
||||||
|
to <code>"newlineAndIndent"</code> by default. You can bind it to
|
||||||
|
something else to get different behavior. To disable special handling
|
||||||
|
completely and only get a newline character inserted, you can bind it
|
||||||
|
to <code>false</code>:</p>
|
||||||
|
|
||||||
|
<pre> extraKeys: {"Enter": false}</pre>
|
||||||
|
|
||||||
|
<p>The same works for <code>Tab</code>. If you don't want CodeMirror
|
||||||
|
to handle it, bind it to <code>false</code>. The default behaviour is
|
||||||
|
to indent the current line more (<code>"indentMore"</code> command),
|
||||||
|
and indent it less when shift is held (<code>"indentLess"</code>).
|
||||||
|
There are also <code>"indentAuto"</code> (smart indent)
|
||||||
|
and <code>"insertTab"</code> commands provided for alternate
|
||||||
|
behaviors. Or you can write your own handler function to do something
|
||||||
|
different altogether.</p>
|
||||||
|
|
||||||
|
<h2>Tabs</h2>
|
||||||
|
|
||||||
|
<p>Handling of tabs changed completely. The display width of tabs can
|
||||||
|
now be set with the <code>tabSize</code> option, and tabs can
|
||||||
|
be <a href="../demo/visibletabs.html">styled</a> by setting CSS rules
|
||||||
|
for the <code>cm-tab</code> class.</p>
|
||||||
|
|
||||||
|
<p>The default width for tabs is now 4, as opposed to the 8 that is
|
||||||
|
hard-wired into browsers. If you are relying on 8-space tabs, make
|
||||||
|
sure you explicitly set <code>tabSize: 8</code> in your options.</p>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
480
codemirror-2.36/index.html
Normal file
480
codemirror-2.36/index.html
Normal file
|
@ -0,0 +1,480 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<title>CodeMirror</title>
|
||||||
|
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Droid+Sans|Droid+Sans:bold"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="doc/docs.css"/>
|
||||||
|
<link rel="alternate" href="https://groups.google.com/group/codemirror-announce/feed/atom_v1_0_topics.xml" type="application/atom+xml"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<h1><span class="logo-braces">{ }</span> <a href="http://codemirror.net/">CodeMirror</a></h1>
|
||||||
|
|
||||||
|
<div class="grey">
|
||||||
|
<img src="doc/baboon.png" class="logo" alt="logo"/>
|
||||||
|
<pre>
|
||||||
|
/* In-browser code editing
|
||||||
|
made bearable */
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="clear"><div class="left blk">
|
||||||
|
|
||||||
|
<p style="margin-top: 0">CodeMirror is a JavaScript component that
|
||||||
|
provides a code editor in the browser. When a mode is available for
|
||||||
|
the language you are coding in, it will color your code, and
|
||||||
|
optionally help with indentation.</p>
|
||||||
|
|
||||||
|
<p>A <a href="doc/manual.html">rich programming API</a> and a CSS
|
||||||
|
theming system are available for customizing CodeMirror to fit your
|
||||||
|
application, and extending it with new functionality.</p>
|
||||||
|
|
||||||
|
<div class="clear"><div class="left1 blk">
|
||||||
|
|
||||||
|
<h2 style="margin-top: 0">Supported modes:</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="mode/clike/index.html">C, C++, C#</a></li>
|
||||||
|
<li><a href="mode/clojure/index.html">Clojure</a></li>
|
||||||
|
<li><a href="mode/coffeescript/index.html">CoffeeScript</a></li>
|
||||||
|
<li><a href="mode/commonlisp/index.html">Common Lisp</a></li>
|
||||||
|
<li><a href="mode/css/index.html">CSS</a></li>
|
||||||
|
<li><a href="mode/diff/index.html">diff</a></li>
|
||||||
|
<li><a href="mode/ecl/index.html">ECL</a></li>
|
||||||
|
<li><a href="mode/erlang/index.html">Erlang</a></li>
|
||||||
|
<li><a href="mode/go/index.html">Go</a></li>
|
||||||
|
<li><a href="mode/groovy/index.html">Groovy</a></li>
|
||||||
|
<li><a href="mode/haskell/index.html">Haskell</a></li>
|
||||||
|
<li><a href="mode/haxe/index.html">Haxe</a></li>
|
||||||
|
<li><a href="mode/htmlembedded/index.html">HTML embedded scripts</a></li>
|
||||||
|
<li><a href="mode/htmlmixed/index.html">HTML mixed-mode</a></li>
|
||||||
|
<li><a href="mode/clike/index.html">Java</a></li>
|
||||||
|
<li><a href="mode/javascript/index.html">JavaScript</a></li>
|
||||||
|
<li><a href="mode/jinja2/index.html">Jinja2</a></li>
|
||||||
|
<li><a href="mode/less/index.html">LESS</a></li>
|
||||||
|
<li><a href="mode/lua/index.html">Lua</a></li>
|
||||||
|
<li><a href="mode/markdown/index.html">Markdown</a> (<a href="mode/gfm/index.html">GitHub-flavour</a>)</li>
|
||||||
|
<li><a href="mode/mysql/index.html">MySQL</a></li>
|
||||||
|
<li><a href="mode/ntriples/index.html">NTriples</a></li>
|
||||||
|
<li><a href="mode/ocaml/index.html">OCaml</a></li>
|
||||||
|
<li><a href="mode/pascal/index.html">Pascal</a></li>
|
||||||
|
<li><a href="mode/perl/index.html">Perl</a></li>
|
||||||
|
<li><a href="mode/php/index.html">PHP</a></li>
|
||||||
|
<li><a href="mode/pig/index.html">Pig Latin</a></li>
|
||||||
|
<li><a href="mode/plsql/index.html">PL/SQL</a></li>
|
||||||
|
<li><a href="mode/properties/index.html">Properties files</a></li>
|
||||||
|
<li><a href="mode/python/index.html">Python</a></li>
|
||||||
|
<li><a href="mode/r/index.html">R</a></li>
|
||||||
|
<li>RPM <a href="mode/rpm/spec/index.html">spec</a> and <a href="mode/rpm/changes/index.html">changelog</a></li>
|
||||||
|
<li><a href="mode/rst/index.html">reStructuredText</a></li>
|
||||||
|
<li><a href="mode/ruby/index.html">Ruby</a></li>
|
||||||
|
<li><a href="mode/rust/index.html">Rust</a></li>
|
||||||
|
<li><a href="mode/clike/scala.html">Scala</a></li>
|
||||||
|
<li><a href="mode/scheme/index.html">Scheme</a></li>
|
||||||
|
<li><a href="mode/shell/index.html">Shell</a></li>
|
||||||
|
<li><a href="mode/sieve/index.html">Sieve</a></li>
|
||||||
|
<li><a href="mode/smalltalk/index.html">Smalltalk</a></li>
|
||||||
|
<li><a href="mode/smarty/index.html">Smarty</a></li>
|
||||||
|
<li><a href="mode/sparql/index.html">SPARQL</a></li>
|
||||||
|
<li><a href="mode/stex/index.html">sTeX, LaTeX</a></li>
|
||||||
|
<li><a href="mode/tiddlywiki/index.html">Tiddlywiki</a></li>
|
||||||
|
<li><a href="mode/tiki/index.html">Tiki wiki</a></li>
|
||||||
|
<li><a href="mode/vb/index.html">VB.NET</a></li>
|
||||||
|
<li><a href="mode/vbscript/index.html">VBScript</a></li>
|
||||||
|
<li><a href="mode/velocity/index.html">Velocity</a></li>
|
||||||
|
<li><a href="mode/verilog/index.html">Verilog</a></li>
|
||||||
|
<li><a href="mode/xml/index.html">XML/HTML</a></li>
|
||||||
|
<li><a href="mode/xquery/index.html">XQuery</a></li>
|
||||||
|
<li><a href="mode/yaml/index.html">YAML</a></li>
|
||||||
|
<li><a href="mode/z80/index.html">Z80</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
</div><div class="left2 blk">
|
||||||
|
|
||||||
|
<h2 style="margin-top: 0">Usage demos:</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="demo/complete.html">Autocompletion</a> (<a href="demo/xmlcomplete.html">XML</a>)</li>
|
||||||
|
<li><a href="demo/search.html">Search/replace</a></li>
|
||||||
|
<li><a href="demo/folding.html">Code folding</a></li>
|
||||||
|
<li><a href="demo/mustache.html">Mode overlays</a></li>
|
||||||
|
<li><a href="demo/multiplex.html">Mode multiplexer</a></li>
|
||||||
|
<li><a href="demo/preview.html">HTML editor with preview</a></li>
|
||||||
|
<li><a href="demo/resize.html">Auto-resizing editor</a></li>
|
||||||
|
<li><a href="demo/marker.html">Setting breakpoints</a></li>
|
||||||
|
<li><a href="demo/activeline.html">Highlighting the current line</a></li>
|
||||||
|
<li><a href="demo/matchhighlighter.html">Highlighting selection matches</a></li>
|
||||||
|
<li><a href="demo/theme.html">Theming</a></li>
|
||||||
|
<li><a href="demo/runmode.html">Stand-alone highlighting</a></li>
|
||||||
|
<li><a href="demo/fullscreen.html">Full-screen editing</a></li>
|
||||||
|
<li><a href="demo/changemode.html">Mode auto-changing</a></li>
|
||||||
|
<li><a href="demo/visibletabs.html">Visible tabs</a></li>
|
||||||
|
<li><a href="demo/formatting.html">Autoformatting of code</a></li>
|
||||||
|
<li><a href="demo/emacs.html">Emacs keybindings</a></li>
|
||||||
|
<li><a href="demo/vim.html">Vim keybindings</a></li>
|
||||||
|
<li><a href="demo/closetag.html">Automatic xml tag closing</a></li>
|
||||||
|
<li><a href="demo/loadmode.html">Lazy mode loading</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h2>Real-world uses:</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="http://jsbin.com">jsbin.com</a> (JS playground)</li>
|
||||||
|
<li><a href="http://www.chris-granger.com/2012/04/12/light-table---a-new-ide-concept/">Light Table</a> (experimental IDE)</li>
|
||||||
|
<li><a href="http://brackets.io">Adobe Brackets</a> (code editor)</li>
|
||||||
|
<li><a href="http://www.mergely.com/">Mergely</a> (interactive diffing)</li>
|
||||||
|
<li><a href="https://script.google.com/">Google Apps Script</a></li>
|
||||||
|
<li><a href="https://github.com/github/android">GitHub's Android app</a></li>
|
||||||
|
<li><a href="http://eloquentjavascript.net/chapter1.html">Eloquent JavaScript</a> (book)</li>
|
||||||
|
<li><a href="http://media.chikuyonok.ru/codemirror2/">Zen Coding</a> (fast XML editing)</li>
|
||||||
|
<li><a href="http://paperjs.org/">Paper.js</a> (graphics scripting)</li>
|
||||||
|
<li><a href="http://tour.golang.org">Go language tour</a></li>
|
||||||
|
<li><a href="http://codev.it/">Codev</a> (collaborative IDE)</li>
|
||||||
|
<li><a href="http://enjalot.com/tributary/2636296/sinwaves.js">Tributary</a> (augmented editing)</li>
|
||||||
|
<li><a href="http://prose.io/">Prose.io</a> (github content editor)</li>
|
||||||
|
<li><a href="http://www.wescheme.org/">WeScheme</a> (learning tool)</li>
|
||||||
|
<li><a href="http://webglplayground.net/">WebGL playground</a></li>
|
||||||
|
<li><a href="http://ql.io/">ql.io</a> (http API query helper)</li>
|
||||||
|
<li><a href="http://elm-lang.org/Examples.elm">Elm language examples</a></li>
|
||||||
|
<li><a href="https://thefiletree.com">The File Tree</a> (collab editor)</li>
|
||||||
|
<li><a href="http://www.jshint.com/">JSHint</a> (JS linter)</li>
|
||||||
|
<li><a href="http://kl1p.com/cmtest/1">kl1p</a> (paste service)</li>
|
||||||
|
<li><a href="http://sqlfiddle.com">SQLFiddle</a> (SQL playground)</li>
|
||||||
|
<li><a href="http://try.haxe.org">Try Haxe</a> (Haxe Playground) </li>
|
||||||
|
<li><a href="http://cssdeck.com/">CSSDeck</a> (CSS showcase)</li>
|
||||||
|
<li><a href="http://www.sketchpatch.net/labs/livecodelabIntro.html">sketchPatch Livecodelab</a></li>
|
||||||
|
<li><a href="https://notex.ch">NoTex</a> (rST authoring)</li>
|
||||||
|
<li><a href="doc/realworld.html">More...</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
</div></div>
|
||||||
|
|
||||||
|
<h2 id="code">Getting the code</h2>
|
||||||
|
|
||||||
|
<p>All of CodeMirror is released under a <a
|
||||||
|
href="LICENSE">MIT-style</a> license. To get it, you can download
|
||||||
|
the <a href="http://codemirror.net/codemirror.zip">latest
|
||||||
|
release</a> or the current <a
|
||||||
|
href="http://codemirror.net/codemirror-latest.zip">development
|
||||||
|
snapshot</a> as zip files. To create a custom minified script file,
|
||||||
|
you can use the <a href="doc/compress.html">compression API</a>.</p>
|
||||||
|
|
||||||
|
<p>We use <a href="http://git-scm.com/">git</a> for version control.
|
||||||
|
The main repository can be fetched in this way:</p>
|
||||||
|
|
||||||
|
<pre class="code">git clone http://marijnhaverbeke.nl/git/codemirror</pre>
|
||||||
|
|
||||||
|
<p>CodeMirror can also be found on GitHub at <a
|
||||||
|
href="http://github.com/marijnh/CodeMirror">marijnh/CodeMirror</a>.
|
||||||
|
If you plan to hack on the code and contribute patches, the best way
|
||||||
|
to do it is to create a GitHub fork, and send pull requests.</p>
|
||||||
|
|
||||||
|
<h2 id="documention">Documentation</h2>
|
||||||
|
|
||||||
|
<p>The <a href="doc/manual.html">manual</a> is your first stop for
|
||||||
|
learning how to use this library. It starts with a quick explanation
|
||||||
|
of how to use the editor, and then describes the API in detail.</p>
|
||||||
|
|
||||||
|
<p>For those who want to learn more about the code, there is
|
||||||
|
an <a href="doc/internals.html">overview of the internals</a> available.
|
||||||
|
The <a href="http://github.com/marijnh/CodeMirror">source code</a>
|
||||||
|
itself is, for the most part, also well commented.</p>
|
||||||
|
|
||||||
|
<h2 id="support">Support and bug reports</h2>
|
||||||
|
|
||||||
|
<p>Community discussion, questions, and informal bug reporting is
|
||||||
|
done on
|
||||||
|
the <a href="http://groups.google.com/group/codemirror">CodeMirror
|
||||||
|
Google group</a>. There is a separate
|
||||||
|
group, <a href="http://groups.google.com/group/codemirror-announce">CodeMirror-announce</a>,
|
||||||
|
which is lower-volume, and is only used for major announcements—new
|
||||||
|
versions and such. These will be cross-posted to both groups, so you
|
||||||
|
don't need to subscribe to both.</p>
|
||||||
|
|
||||||
|
<p>Though bug reports through e-mail are responded to, the preferred
|
||||||
|
way to report bugs is to use
|
||||||
|
the <a href="http://github.com/marijnh/CodeMirror/issues">GitHub
|
||||||
|
issue tracker</a>. Before reporting a
|
||||||
|
bug, <a href="doc/reporting.html">read these pointers</a>. Also,
|
||||||
|
the issue tracker is for <em>bugs</em>, not requests for help.</p>
|
||||||
|
|
||||||
|
<p>When none of these seem fitting, you can
|
||||||
|
simply <a href="mailto:marijnh@gmail.com">e-mail the maintainer</a>
|
||||||
|
directly.</p>
|
||||||
|
|
||||||
|
<h2 id="supported">Supported browsers</h2>
|
||||||
|
|
||||||
|
<p>The following <em>desktop</em> browsers are able to run CodeMirror:</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li>Firefox 2 or higher</li>
|
||||||
|
<li>Chrome, any version</li>
|
||||||
|
<li>Safari 3 or higher</li>
|
||||||
|
<li>Opera 9 or higher (with some key-handling problems on OS X)</li>
|
||||||
|
<li>Internet Explorer 7 or higher in standards mode<br>
|
||||||
|
<em>(So not quirks mode. But quasi-standards mode with a
|
||||||
|
transitional doctype is also flaky. <code><!doctype
|
||||||
|
html></code> is recommended.)</em></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p>I am not actively testing against every new browser release, and
|
||||||
|
vendors have a habit of introducing bugs all the time, so I am
|
||||||
|
relying on the community to tell me when something breaks.
|
||||||
|
See <a href="#support">here</a> for information on how to contact
|
||||||
|
me.</p>
|
||||||
|
|
||||||
|
<p>Mobile browsers mostly kind of work, but, because of limitations
|
||||||
|
and their fundamentally different UI assumptions, show a lot of
|
||||||
|
quirks that are hard to work around.</p>
|
||||||
|
|
||||||
|
<h2 id="commercial">Commercial support</h2>
|
||||||
|
|
||||||
|
<p>CodeMirror is developed and maintained by me, Marijn Haverbeke,
|
||||||
|
in my own time. If your company is getting value out of CodeMirror,
|
||||||
|
please consider purchasing a support contract.</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li>You'll be funding further work on CodeMirror.</li>
|
||||||
|
<li>You ensure that you get a quick response when you have a
|
||||||
|
problem, even when I am otherwise busy.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p>CodeMirror support contracts exist in two
|
||||||
|
forms—<strong>basic</strong> at €100 per month,
|
||||||
|
and <strong>premium</strong> at €500 per
|
||||||
|
month. <a href="mailto:marijnh@gmail.com">Contact me</a> for further
|
||||||
|
information.</p>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="right blk">
|
||||||
|
|
||||||
|
<a href="http://codemirror.net/codemirror.zip" class="download">Download the latest release</a>
|
||||||
|
|
||||||
|
<h2>Support CodeMirror</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li>Donate
|
||||||
|
(<span onclick="document.getElementById('paypal').submit();"
|
||||||
|
class="quasilink">Paypal</span>,
|
||||||
|
<span onclick="document.getElementById('bankinfo').style.display = 'block';"
|
||||||
|
class="quasilink">bank</span>, or
|
||||||
|
<a href="https://www.gittip.com/marijnh">Gittip</a>)</li>
|
||||||
|
<li>Purchase <a href="#commercial">commercial support</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p id="bankinfo" style="display: none;">
|
||||||
|
Bank: <i>Rabobank</i><br/>
|
||||||
|
Country: <i>Netherlands</i><br/>
|
||||||
|
SWIFT: <i>RABONL2U</i><br/>
|
||||||
|
Account: <i>147850770</i><br/>
|
||||||
|
Name: <i>Marijn Haverbeke</i><br/>
|
||||||
|
IBAN: <i>NL26 RABO 0147 8507 70</i>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<h2>Reading material</h2>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><a href="doc/manual.html">User manual</a></li>
|
||||||
|
<li><a href="http://github.com/marijnh/CodeMirror">Browse the code</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h2 id=releases>Releases</h2>
|
||||||
|
|
||||||
|
<p class="rel">20-11-2012: <a href="http://codemirror.net/codemirror-2.36.zip">Version 2.36</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New mode: <a href="mode/z80/index.html">Z80 assembly</a>.</li>
|
||||||
|
<li>New theme: <a href="demo/theme.html?twilight">Twilight</a>.</li>
|
||||||
|
<li>Add command-line compression helper.</li>
|
||||||
|
<li>Make <a href="doc/manual.html#scrollIntoView"><code>scrollIntoView</code></a> public.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#defaultTextHeight"><code>defaultTextHeight</code></a> method.</li>
|
||||||
|
<li>Various extensions to the vim keymap.</li>
|
||||||
|
<li>Make <a href="mode/php/index.html">PHP mode</a> build on <a href="mode/htmlmixed/index.html">mixed HTML mode</a>.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#util_continuecomment">comment-continuing</a> add-on.</li>
|
||||||
|
<li>Full <a href="https://github.com/marijnh/CodeMirror/compare/v2.35...v2.36">list of patches</a>.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">20-11-2012: <a href="http://codemirror.net/codemirror-3.0rc1.zip">Version 3.0, release candidate 1</a>:</p>
|
||||||
|
|
||||||
|
<p class="rel-note"><strong>New major version</strong>. Only partially
|
||||||
|
backwards-compatible. See
|
||||||
|
the <a href="http://codemirror.net/3/doc/upgrade_v3.html">upgrading
|
||||||
|
guide</a> for more information. Changes since beta 2:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New theme: <a href="http://codemirror.net/3/demo/theme.html?solarized%20light">Solarized</a>.</li>
|
||||||
|
<li>Introduce <a href="http://codemirror.net/3/doc/manual.html#addLineClass"><code>addLineClass</code></a>
|
||||||
|
and <a href="http://codemirror.net/3/doc/manual.html#removeLineClass"><code>removeLineClass</code></a>,
|
||||||
|
drop <code>setLineClass</code>.</li>
|
||||||
|
<li>Add a <em>lot</em> of
|
||||||
|
new <a href="http://codemirror.net/3/doc/manual.html#markText">options for marked text</a>
|
||||||
|
(read-only, atomic, collapsed, widget replacement).</li>
|
||||||
|
<li>Remove the old code folding interface in favour of these new ranges.</li>
|
||||||
|
<li>Add <a href="http://codemirror.net/3/doc/manual.html#isClean"><code>isClean</code></a>/<a href="http://codemirror.net/3/doc/manual.html#markClean"><code>markClean</code></a> methods.</li>
|
||||||
|
<li>Remove <code>compoundChange</code> method, use better undo-event-combining heuristic.</li>
|
||||||
|
<li>Improve scrolling performance smoothness.</li>
|
||||||
|
<li>Full <a href="https://github.com/marijnh/CodeMirror/compare/v3.0beta2...v3.0rc1">list of patches</a>.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">22-10-2012: <a href="http://codemirror.net/codemirror-2.35.zip">Version 2.35</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New (sub) mode: <a href="mode/javascript/typescript.html">TypeScript</a>.</li>
|
||||||
|
<li>Don't overwrite (insert key) when pasting.</li>
|
||||||
|
<li>Fix several bugs in <a href="doc/manual.html#markText"><code>markText</code></a>/undo interaction.</li>
|
||||||
|
<li>Better indentation of JavaScript code without semicolons.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#defineInitHook"><code>defineInitHook</code></a> function.</li>
|
||||||
|
<li>Full <a href="https://github.com/marijnh/CodeMirror/compare/v2.34...v2.35">list of patches</a>.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">22-10-2012: <a href="http://codemirror.net/codemirror-3.0beta2.zip">Version 3.0, beta 2</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Fix page-based coordinate computation.</li>
|
||||||
|
<li>Fix firing of <a href="http://codemirror.net/3/doc/manual.html#event_gutterClick"><code>gutterClick</code></a> event.</li>
|
||||||
|
<li>Add <a href="http://codemirror.net/3/doc/manual.html#option_cursorHeight"><code>cursorHeight</code></a> option.</li>
|
||||||
|
<li>Fix bi-directional text regression.</li>
|
||||||
|
<li>Add <a href="http://codemirror.net/3/doc/manual.html#option_viewportMargin"><code>viewportMargin</code></a> option.</li>
|
||||||
|
<li>Directly handle mousewheel events (again, hopefully better).</li>
|
||||||
|
<li>Make vertical cursor movement more robust (through widgets, big line gaps).</li>
|
||||||
|
<li>Add <a href="http://codemirror.net/3/doc/manual.html#option_flattenSpans"><code>flattenSpans</code></a> option.</li>
|
||||||
|
<li>Initialization in hidden state works again.</li>
|
||||||
|
<li>Many optimizations. Poor responsiveness should be fixed.</li>
|
||||||
|
<li>Full <a href="https://github.com/marijnh/CodeMirror/compare/v3.0beta1...v3.0beta2">list of patches</a>.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">19-09-2012: <a href="http://codemirror.net/codemirror-2.34.zip">Version 2.34</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New mode: <a href="mode/commonlisp/index.html">Common Lisp</a>.</li>
|
||||||
|
<li>Fix right-click select-all on most browsers.</li>
|
||||||
|
<li>Change the way highlighting happens:<br> Saves memory and CPU cycles.<br> <code>compareStates</code> is no longer needed.<br> <code>onHighlightComplete</code> no longer works.</li>
|
||||||
|
<li>Integrate mode (Markdown, XQuery, CSS, sTex) tests in central testsuite.</li>
|
||||||
|
<li>Add a <a href="doc/manual.html#version"><code>CodeMirror.version</code></a> property.</li>
|
||||||
|
<li>More robust handling of nested modes in <a href="demo/formatting.html">formatting</a> and <a href="demo/closetag.html">closetag</a> plug-ins.</li>
|
||||||
|
<li>Un/redo now preserves <a href="doc/manual.html#markText">marked text</a> and bookmarks.</li>
|
||||||
|
<li><a href="https://github.com/marijnh/CodeMirror/compare/v2.33...v2.34">Full list</a> of patches.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">19-09-2012: <a href="http://codemirror.net/codemirror-3.0beta1.zip">Version 3.0, beta 1</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Bi-directional text support.</li>
|
||||||
|
<li>More powerful gutter model.</li>
|
||||||
|
<li>Support for arbitrary text/widget height.</li>
|
||||||
|
<li>In-line widgets.</li>
|
||||||
|
<li>Generalized event handling.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">23-08-2012: <a href="http://codemirror.net/codemirror-2.33.zip">Version 2.33</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New mode: <a href="mode/sieve/index.html">Sieve</a>.</li>
|
||||||
|
<li>New <a href="doc/manual.html#getViewport"><code>getViewPort</code></a> and <a href="doc/manual.html#option_onViewportChange"><code>onViewportChange</code></a> API.</li>
|
||||||
|
<li><a href="doc/manual.html#option_cursorBlinkRate">Configurable</a> cursor blink rate.</li>
|
||||||
|
<li>Make binding a key to <code>false</code> disabling handling (again).</li>
|
||||||
|
<li>Show non-printing characters as red dots.</li>
|
||||||
|
<li>More tweaks to the scrolling model.</li>
|
||||||
|
<li>Expanded testsuite. Basic linter added.</li>
|
||||||
|
<li>Remove most uses of <code>innerHTML</code>. Remove <code>CodeMirror.htmlEscape</code>.</li>
|
||||||
|
<li><a href="https://github.com/marijnh/CodeMirror/compare/v2.32...v2.33">Full list</a> of patches.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">23-07-2012: <a href="http://codemirror.net/codemirror-2.32.zip">Version 2.32</a>:</p>
|
||||||
|
|
||||||
|
<p class="rel-note">Emergency fix for a bug where an editor with
|
||||||
|
line wrapping on IE will break when there is <em>no</em>
|
||||||
|
scrollbar.</p>
|
||||||
|
|
||||||
|
<p class="rel">20-07-2012: <a href="http://codemirror.net/codemirror-2.31.zip">Version 2.31</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New modes: <a href="mode/ocaml/index.html">OCaml</a>, <a href="mode/haxe/index.html">Haxe</a>, and <a href="mode/vb/index.html">VB.NET</a>.</li>
|
||||||
|
<li>Several fixes to the new scrolling model.</li>
|
||||||
|
<li>Add a <a href="doc/manual.html#setSize"><code>setSize</code></a> method for programmatic resizing.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#getHistory"><code>getHistory</code></a> and <a href="doc/manual.html#setHistory"><code>setHistory</code></a> methods.</li>
|
||||||
|
<li>Allow custom line separator string in <a href="doc/manual.html#getValue"><code>getValue</code></a> and <a href="doc/manual.html#getRange"><code>getRange</code></a>.</li>
|
||||||
|
<li>Support double- and triple-click drag, double-clicking whitespace.</li>
|
||||||
|
<li>And more... <a href="https://github.com/marijnh/CodeMirror/compare/v2.3...v2.31">(all patches)</a></li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">22-06-2012: <a href="http://codemirror.net/codemirror-2.3.zip">Version 2.3</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li><strong>New scrollbar implementation</strong>. Should flicker less. Changes DOM structure of the editor.</li>
|
||||||
|
<li>New theme: <a href="demo/theme.html?vibrant-ink">vibrant-ink</a>.</li>
|
||||||
|
<li>Many extensions to the VIM keymap (including text objects).</li>
|
||||||
|
<li>Add <a href="demo/multiplex.html">mode-multiplexing</a> utility script.</li>
|
||||||
|
<li>Fix bug where right-click paste works in read-only mode.</li>
|
||||||
|
<li>Add a <a href="doc/manual.html#getScrollInfo"><code>getScrollInfo</code></a> method.</li>
|
||||||
|
<li>Lots of other <a href="https://github.com/marijnh/CodeMirror/compare/v2.25...v2.3">fixes</a>.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">23-05-2012: <a href="http://codemirror.net/codemirror-2.25.zip">Version 2.25</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>New mode: <a href="mode/erlang/index.html">Erlang</a>.</li>
|
||||||
|
<li><strong>Remove xmlpure mode</strong> (use <a href="mode/xml/index.html">xml.js</a>).</li>
|
||||||
|
<li>Fix line-wrapping in Opera.</li>
|
||||||
|
<li>Fix X Windows middle-click paste in Chrome.</li>
|
||||||
|
<li>Fix bug that broke pasting of huge documents.</li>
|
||||||
|
<li>Fix backspace and tab key repeat in Opera.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">23-04-2012: <a href="http://codemirror.net/codemirror-2.24.zip">Version 2.24</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li><strong>Drop support for Internet Explorer 6</strong>.</li>
|
||||||
|
<li>New
|
||||||
|
modes: <a href="mode/shell/index.html">Shell</a>, <a href="mode/tiki/index.html">Tiki
|
||||||
|
wiki</a>, <a href="mode/pig/index.html">Pig Latin</a>.</li>
|
||||||
|
<li>New themes: <a href="demo/theme.html?ambiance">Ambiance</a>, <a href="demo/theme.html?blackboard">Blackboard</a>.</li>
|
||||||
|
<li>More control over drag/drop
|
||||||
|
with <a href="doc/manual.html#option_dragDrop"><code>dragDrop</code></a>
|
||||||
|
and <a href="doc/manual.html#option_onDragEvent"><code>onDragEvent</code></a>
|
||||||
|
options.</li>
|
||||||
|
<li>Make HTML mode a bit less pedantic.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#compoundChange"><code>compoundChange</code></a> API method.</li>
|
||||||
|
<li>Several fixes in undo history and line hiding.</li>
|
||||||
|
<li>Remove (broken) support for <code>catchall</code> in key maps,
|
||||||
|
add <code>nofallthrough</code> boolean field instead.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p class="rel">26-03-2012: <a href="http://codemirror.net/codemirror-2.23.zip">Version 2.23</a>:</p>
|
||||||
|
|
||||||
|
<ul class="rel-note">
|
||||||
|
<li>Change <strong>default binding for tab</strong> <a href="javascript:void(document.getElementById('tabbinding').style.display='')">[more]</a>
|
||||||
|
<div style="display: none" id=tabbinding>
|
||||||
|
Starting in 2.23, these bindings are default:
|
||||||
|
<ul><li>Tab: Insert tab character</li>
|
||||||
|
<li>Shift-tab: Reset line indentation to default</li>
|
||||||
|
<li>Ctrl/Cmd-[: Reduce line indentation (old tab behaviour)</li>
|
||||||
|
<li>Ctrl/Cmd-]: Increase line indentation (old shift-tab behaviour)</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<li>New modes: <a href="mode/xquery/index.html">XQuery</a> and <a href="mode/vbscript/index.html">VBScript</a>.</li>
|
||||||
|
<li>Two new themes: <a href="mode/less/index.html">lesser-dark</a> and <a href="mode/xquery/index.html">xq-dark</a>.</li>
|
||||||
|
<li>Differentiate between background and text styles in <a href="doc/manual.html#setLineClass"><code>setLineClass</code></a>.</li>
|
||||||
|
<li>Fix drag-and-drop in IE9+.</li>
|
||||||
|
<li>Extend <a href="doc/manual.html#charCoords"><code>charCoords</code></a>
|
||||||
|
and <a href="doc/manual.html#cursorCoords"><code>cursorCoords</code></a> with a <code>mode</code> argument.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#option_autofocus"><code>autofocus</code></a> option.</li>
|
||||||
|
<li>Add <a href="doc/manual.html#findMarksAt"><code>findMarksAt</code></a> method.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p><a href="doc/oldrelease.html">Older releases...</a></p>
|
||||||
|
|
||||||
|
</div></div>
|
||||||
|
|
||||||
|
<div style="height: 2em"> </div>
|
||||||
|
|
||||||
|
<form action="https://www.paypal.com/cgi-bin/webscr" method="post" id="paypal">
|
||||||
|
<input type="hidden" name="cmd" value="_s-xclick"/>
|
||||||
|
<input type="hidden" name="hosted_button_id" value="3FVHS5FGUY7CC"/>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
30
codemirror-2.36/keymap/emacs.js
Normal file
30
codemirror-2.36/keymap/emacs.js
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
// TODO number prefixes
|
||||||
|
(function() {
|
||||||
|
// Really primitive kill-ring implementation.
|
||||||
|
var killRing = [];
|
||||||
|
function addToRing(str) {
|
||||||
|
killRing.push(str);
|
||||||
|
if (killRing.length > 50) killRing.shift();
|
||||||
|
}
|
||||||
|
function getFromRing() { return killRing[killRing.length - 1] || ""; }
|
||||||
|
function popFromRing() { if (killRing.length > 1) killRing.pop(); return getFromRing(); }
|
||||||
|
|
||||||
|
CodeMirror.keyMap.emacs = {
|
||||||
|
"Ctrl-X": function(cm) {cm.setOption("keyMap", "emacs-Ctrl-X");},
|
||||||
|
"Ctrl-W": function(cm) {addToRing(cm.getSelection()); cm.replaceSelection("");},
|
||||||
|
"Ctrl-Alt-W": function(cm) {addToRing(cm.getSelection()); cm.replaceSelection("");},
|
||||||
|
"Alt-W": function(cm) {addToRing(cm.getSelection());},
|
||||||
|
"Ctrl-Y": function(cm) {cm.replaceSelection(getFromRing());},
|
||||||
|
"Alt-Y": function(cm) {cm.replaceSelection(popFromRing());},
|
||||||
|
"Ctrl-/": "undo", "Shift-Ctrl--": "undo", "Shift-Alt-,": "goDocStart", "Shift-Alt-.": "goDocEnd",
|
||||||
|
"Ctrl-S": "findNext", "Ctrl-R": "findPrev", "Ctrl-G": "clearSearch", "Shift-Alt-5": "replace",
|
||||||
|
"Ctrl-Z": "undo", "Cmd-Z": "undo", "Alt-/": "autocomplete", "Alt-V": "goPageUp",
|
||||||
|
"Ctrl-J": "newlineAndIndent", "Enter": false, "Tab": "indentAuto",
|
||||||
|
fallthrough: ["basic", "emacsy"]
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap["emacs-Ctrl-X"] = {
|
||||||
|
"Ctrl-S": "save", "Ctrl-W": "save", "S": "saveAll", "F": "open", "U": "undo", "K": "close",
|
||||||
|
auto: "emacs", nofallthrough: true
|
||||||
|
};
|
||||||
|
})();
|
897
codemirror-2.36/keymap/vim.js
Normal file
897
codemirror-2.36/keymap/vim.js
Normal file
|
@ -0,0 +1,897 @@
|
||||||
|
// Supported keybindings:
|
||||||
|
//
|
||||||
|
// Cursor movement:
|
||||||
|
// h, j, k, l
|
||||||
|
// e, E, w, W, b, B
|
||||||
|
// Ctrl-f, Ctrl-b
|
||||||
|
// Ctrl-n, Ctrl-p
|
||||||
|
// $, ^, 0
|
||||||
|
// G
|
||||||
|
// ge, gE
|
||||||
|
// gg
|
||||||
|
// f<char>, F<char>, t<char>, T<char>
|
||||||
|
// Ctrl-o, Ctrl-i TODO (FIXME - Ctrl-O wont work in Chrome)
|
||||||
|
// /, ?, n, N TODO (does not work)
|
||||||
|
// #, * TODO
|
||||||
|
//
|
||||||
|
// Entering insert mode:
|
||||||
|
// i, I, a, A, o, O
|
||||||
|
// s
|
||||||
|
// ce, cb
|
||||||
|
// cc
|
||||||
|
// S, C TODO
|
||||||
|
// cf<char>, cF<char>, ct<char>, cT<char>
|
||||||
|
//
|
||||||
|
// Deleting text:
|
||||||
|
// x, X
|
||||||
|
// J
|
||||||
|
// dd, D
|
||||||
|
// de, db
|
||||||
|
// df<char>, dF<char>, dt<char>, dT<char>
|
||||||
|
//
|
||||||
|
// Yanking and pasting:
|
||||||
|
// yy, Y
|
||||||
|
// p, P
|
||||||
|
// p'<char> TODO - test
|
||||||
|
// y'<char> TODO - test
|
||||||
|
// m<char> TODO - test
|
||||||
|
//
|
||||||
|
// Changing text in place:
|
||||||
|
// ~
|
||||||
|
// r<char>
|
||||||
|
//
|
||||||
|
// Visual mode:
|
||||||
|
// v, V TODO
|
||||||
|
//
|
||||||
|
// Misc:
|
||||||
|
// . TODO
|
||||||
|
//
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
var sdir = "f";
|
||||||
|
var buf = "";
|
||||||
|
var mark = {};
|
||||||
|
var repeatCount = 0;
|
||||||
|
function isLine(cm, line) { return line >= 0 && line < cm.lineCount(); }
|
||||||
|
function emptyBuffer() { buf = ""; }
|
||||||
|
function pushInBuffer(str) { buf += str; }
|
||||||
|
function pushRepeatCountDigit(digit) {return function(cm) {repeatCount = (repeatCount * 10) + digit}; }
|
||||||
|
function getCountOrOne() {
|
||||||
|
var i = repeatCount;
|
||||||
|
return i || 1;
|
||||||
|
}
|
||||||
|
function clearCount() {
|
||||||
|
repeatCount = 0;
|
||||||
|
}
|
||||||
|
function iterTimes(func) {
|
||||||
|
for (var i = 0, c = getCountOrOne(); i < c; ++i) func(i, i == c - 1);
|
||||||
|
clearCount();
|
||||||
|
}
|
||||||
|
function countTimes(func) {
|
||||||
|
if (typeof func == "string") func = CodeMirror.commands[func];
|
||||||
|
return function(cm) { iterTimes(function (i, last) { func(cm, i, last); }); };
|
||||||
|
}
|
||||||
|
|
||||||
|
function iterObj(o, f) {
|
||||||
|
for (var prop in o) if (o.hasOwnProperty(prop)) f(prop, o[prop]);
|
||||||
|
}
|
||||||
|
function iterList(l, f) {
|
||||||
|
for (var i = 0; i < l.length; ++i) f(l[i]);
|
||||||
|
}
|
||||||
|
function toLetter(ch) {
|
||||||
|
// T -> t, Shift-T -> T, '*' -> *, "Space" -> " "
|
||||||
|
if (ch.slice(0, 6) == "Shift-") {
|
||||||
|
return ch.slice(0, 1);
|
||||||
|
} else {
|
||||||
|
if (ch == "Space") return " ";
|
||||||
|
if (ch.length == 3 && ch[0] == "'" && ch[2] == "'") return ch[1];
|
||||||
|
return ch.toLowerCase();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var SPECIAL_SYMBOLS = "~`!@#$%^&*()_-+=[{}]\\|/?.,<>:;\"\'1234567890";
|
||||||
|
function toCombo(ch) {
|
||||||
|
// t -> T, T -> Shift-T, * -> '*', " " -> "Space"
|
||||||
|
if (ch == " ") return "Space";
|
||||||
|
var specialIdx = SPECIAL_SYMBOLS.indexOf(ch);
|
||||||
|
if (specialIdx != -1) return "'" + ch + "'";
|
||||||
|
if (ch.toLowerCase() == ch) return ch.toUpperCase();
|
||||||
|
return "Shift-" + ch.toUpperCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
var word = [/\w/, /[^\w\s]/], bigWord = [/\S/];
|
||||||
|
// Finds a word on the given line, and continue searching the next line if it can't find one.
|
||||||
|
function findWord(cm, lineNum, pos, dir, regexps) {
|
||||||
|
var line = cm.getLine(lineNum);
|
||||||
|
while (true) {
|
||||||
|
var stop = (dir > 0) ? line.length : -1;
|
||||||
|
var wordStart = stop, wordEnd = stop;
|
||||||
|
// Find bounds of next word.
|
||||||
|
for (; pos != stop; pos += dir) {
|
||||||
|
for (var i = 0; i < regexps.length; ++i) {
|
||||||
|
if (regexps[i].test(line.charAt(pos))) {
|
||||||
|
wordStart = pos;
|
||||||
|
// Advance to end of word.
|
||||||
|
for (; pos != stop && regexps[i].test(line.charAt(pos)); pos += dir) {}
|
||||||
|
wordEnd = (dir > 0) ? pos : pos + 1;
|
||||||
|
return {
|
||||||
|
from: Math.min(wordStart, wordEnd),
|
||||||
|
to: Math.max(wordStart, wordEnd),
|
||||||
|
line: lineNum};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Advance to next/prev line.
|
||||||
|
lineNum += dir;
|
||||||
|
if (!isLine(cm, lineNum)) return null;
|
||||||
|
line = cm.getLine(lineNum);
|
||||||
|
pos = (dir > 0) ? 0 : line.length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* @param {boolean} cm CodeMirror object.
|
||||||
|
* @param {regexp} regexps Regular expressions for word characters.
|
||||||
|
* @param {number} dir Direction, +/- 1.
|
||||||
|
* @param {number} times Number of times to advance word.
|
||||||
|
* @param {string} where Go to "start" or "end" of word, 'e' vs 'w'.
|
||||||
|
* @param {boolean} yank Whether we are finding words to yank. If true,
|
||||||
|
* do not go to the next line to look for the last word. This is to
|
||||||
|
* prevent deleting new line on 'dw' at the end of a line.
|
||||||
|
*/
|
||||||
|
function moveToWord(cm, regexps, dir, times, where, yank) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
if (yank) {
|
||||||
|
where = 'start';
|
||||||
|
}
|
||||||
|
for (var i = 0; i < times; i++) {
|
||||||
|
var startCh = cur.ch, startLine = cur.line, word;
|
||||||
|
while (true) {
|
||||||
|
// Search and advance.
|
||||||
|
word = findWord(cm, cur.line, cur.ch, dir, regexps);
|
||||||
|
if (word) {
|
||||||
|
if (yank && times == 1 && dir == 1 && cur.line != word.line) {
|
||||||
|
// Stop at end of line of last word. Don't want to delete line return
|
||||||
|
// for dw if the last deleted word is at the end of a line.
|
||||||
|
cur.ch = cm.getLine(cur.line).length;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
// Move to the word we just found. If by moving to the word we end up
|
||||||
|
// in the same spot, then move an extra character and search again.
|
||||||
|
cur.line = word.line;
|
||||||
|
if (dir > 0 && where == 'end') {
|
||||||
|
// 'e'
|
||||||
|
if (startCh != word.to - 1 || startLine != word.line) {
|
||||||
|
cur.ch = word.to - 1;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
cur.ch = word.to;
|
||||||
|
}
|
||||||
|
} else if (dir > 0 && where == 'start') {
|
||||||
|
// 'w'
|
||||||
|
if (startCh != word.from || startLine != word.line) {
|
||||||
|
cur.ch = word.from;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
cur.ch = word.to;
|
||||||
|
}
|
||||||
|
} else if (dir < 0 && where == 'end') {
|
||||||
|
// 'ge'
|
||||||
|
if (startCh != word.to || startLine != word.line) {
|
||||||
|
cur.ch = word.to;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
cur.ch = word.from - 1;
|
||||||
|
}
|
||||||
|
} else if (dir < 0 && where == 'start') {
|
||||||
|
// 'b'
|
||||||
|
if (startCh != word.from || startLine != word.line) {
|
||||||
|
cur.ch = word.from;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
cur.ch = word.from - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No more words to be found. Move to end of document.
|
||||||
|
for (; isLine(cm, cur.line + dir); cur.line += dir) {}
|
||||||
|
cur.ch = (dir > 0) ? cm.getLine(cur.line).length : 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (where == 'end' && yank) {
|
||||||
|
// Include the last character of the word for actions.
|
||||||
|
cur.ch++;
|
||||||
|
}
|
||||||
|
return cur;
|
||||||
|
}
|
||||||
|
function joinLineNext(cm) {
|
||||||
|
var cur = cm.getCursor(), ch = cur.ch, line = cm.getLine(cur.line);
|
||||||
|
CodeMirror.commands.goLineEnd(cm);
|
||||||
|
if (cur.line != cm.lineCount()) {
|
||||||
|
CodeMirror.commands.goLineEnd(cm);
|
||||||
|
cm.replaceSelection(" ", "end");
|
||||||
|
CodeMirror.commands.delCharRight(cm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function delTillMark(cm, cHar) {
|
||||||
|
var i = mark[cHar];
|
||||||
|
if (i === undefined) {
|
||||||
|
// console.log("Mark not set"); // TODO - show in status bar
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var l = cm.getCursor().line, start = i > l ? l : i, end = i > l ? i : l;
|
||||||
|
cm.setCursor(start);
|
||||||
|
for (var c = start; c <= end; c++) {
|
||||||
|
pushInBuffer("\n" + cm.getLine(start));
|
||||||
|
cm.removeLine(start);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function yankTillMark(cm, cHar) {
|
||||||
|
var i = mark[cHar];
|
||||||
|
if (i === undefined) {
|
||||||
|
// console.log("Mark not set"); // TODO - show in status bar
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var l = cm.getCursor().line, start = i > l ? l : i, end = i > l ? i : l;
|
||||||
|
for (var c = start; c <= end; c++) {
|
||||||
|
pushInBuffer("\n" + cm.getLine(c));
|
||||||
|
}
|
||||||
|
cm.setCursor(start);
|
||||||
|
}
|
||||||
|
function goLineStartText(cm) {
|
||||||
|
// Go to the start of the line where the text begins, or the end for whitespace-only lines
|
||||||
|
var cur = cm.getCursor(), firstNonWS = cm.getLine(cur.line).search(/\S/);
|
||||||
|
cm.setCursor(cur.line, firstNonWS == -1 ? line.length : firstNonWS, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function charIdxInLine(cm, cHar, motion_options) {
|
||||||
|
// Search for cHar in line.
|
||||||
|
// motion_options: {forward, inclusive}
|
||||||
|
// If inclusive = true, include it too.
|
||||||
|
// If forward = true, search forward, else search backwards.
|
||||||
|
// If char is not found on this line, do nothing
|
||||||
|
var cur = cm.getCursor(), line = cm.getLine(cur.line), idx;
|
||||||
|
var ch = toLetter(cHar), mo = motion_options;
|
||||||
|
if (mo.forward) {
|
||||||
|
idx = line.indexOf(ch, cur.ch + 1);
|
||||||
|
if (idx != -1 && mo.inclusive) idx += 1;
|
||||||
|
} else {
|
||||||
|
idx = line.lastIndexOf(ch, cur.ch);
|
||||||
|
if (idx != -1 && !mo.inclusive) idx += 1;
|
||||||
|
}
|
||||||
|
return idx;
|
||||||
|
}
|
||||||
|
|
||||||
|
function moveTillChar(cm, cHar, motion_options) {
|
||||||
|
// Move to cHar in line, as found by charIdxInLine.
|
||||||
|
var idx = charIdxInLine(cm, cHar, motion_options), cur = cm.getCursor();
|
||||||
|
if (idx != -1) cm.setCursor({line: cur.line, ch: idx});
|
||||||
|
}
|
||||||
|
|
||||||
|
function delTillChar(cm, cHar, motion_options) {
|
||||||
|
// delete text in this line, untill cHar is met,
|
||||||
|
// as found by charIdxInLine.
|
||||||
|
// If char is not found on this line, do nothing
|
||||||
|
var idx = charIdxInLine(cm, cHar, motion_options);
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
if (idx !== -1) {
|
||||||
|
if (motion_options.forward) {
|
||||||
|
cm.replaceRange("", {line: cur.line, ch: cur.ch}, {line: cur.line, ch: idx});
|
||||||
|
} else {
|
||||||
|
cm.replaceRange("", {line: cur.line, ch: idx}, {line: cur.line, ch: cur.ch});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function enterInsertMode(cm) {
|
||||||
|
// enter insert mode: switch mode and cursor
|
||||||
|
clearCount();
|
||||||
|
cm.setOption("keyMap", "vim-insert");
|
||||||
|
}
|
||||||
|
|
||||||
|
function dialog(cm, text, shortText, f) {
|
||||||
|
if (cm.openDialog) cm.openDialog(text, f);
|
||||||
|
else f(prompt(shortText, ""));
|
||||||
|
}
|
||||||
|
function showAlert(cm, text) {
|
||||||
|
var esc = text.replace(/[<&]/, function(ch) { return ch == "<" ? "<" : "&"; });
|
||||||
|
if (cm.openDialog) cm.openDialog(esc + " <button type=button>OK</button>");
|
||||||
|
else alert(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
// main keymap
|
||||||
|
var map = CodeMirror.keyMap.vim = {
|
||||||
|
// Pipe (|); TODO: should be *screen* chars, so need a util function to turn tabs into spaces?
|
||||||
|
"'|'": function(cm) {
|
||||||
|
cm.setCursor(cm.getCursor().line, getCountOrOne() - 1, true);
|
||||||
|
clearCount();
|
||||||
|
},
|
||||||
|
"A": function(cm) {
|
||||||
|
cm.setCursor(cm.getCursor().line, cm.getCursor().ch+1, true);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
"Shift-A": function(cm) { CodeMirror.commands.goLineEnd(cm); enterInsertMode(cm);},
|
||||||
|
"I": function(cm) { enterInsertMode(cm);},
|
||||||
|
"Shift-I": function(cm) { goLineStartText(cm); enterInsertMode(cm);},
|
||||||
|
"O": function(cm) {
|
||||||
|
CodeMirror.commands.goLineEnd(cm);
|
||||||
|
CodeMirror.commands.newlineAndIndent(cm);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
"Shift-O": function(cm) {
|
||||||
|
CodeMirror.commands.goLineStart(cm);
|
||||||
|
cm.replaceSelection("\n", "start");
|
||||||
|
cm.indentLine(cm.getCursor().line);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
"G": function(cm) { cm.setOption("keyMap", "vim-prefix-g");},
|
||||||
|
"Shift-D": function(cm) {
|
||||||
|
var cursor = cm.getCursor();
|
||||||
|
var lineN = cursor.line;
|
||||||
|
var line = cm.getLine(lineN);
|
||||||
|
cm.setLine(lineN, line.slice(0, cursor.ch));
|
||||||
|
|
||||||
|
emptyBuffer();
|
||||||
|
pushInBuffer(line.slice(cursor.ch));
|
||||||
|
|
||||||
|
if (repeatCount > 1) {
|
||||||
|
// we've already done it once
|
||||||
|
--repeatCount;
|
||||||
|
// the lines dissapear (ie, cursor stays on the same lineN),
|
||||||
|
// so only incremenet once
|
||||||
|
++lineN;
|
||||||
|
|
||||||
|
iterTimes(function() {
|
||||||
|
pushInBuffer(cm.getLine(lineN));
|
||||||
|
cm.removeLine(lineN);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
"S": function (cm) {
|
||||||
|
countTimes(function (_cm) {
|
||||||
|
CodeMirror.commands.delCharRight(_cm);
|
||||||
|
})(cm);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
"M": function(cm) {cm.setOption("keyMap", "vim-prefix-m"); mark = {};},
|
||||||
|
"Y": function(cm) {cm.setOption("keyMap", "vim-prefix-y"); emptyBuffer();},
|
||||||
|
"Shift-Y": function(cm) {
|
||||||
|
emptyBuffer();
|
||||||
|
iterTimes(function(i) { pushInBuffer("\n" + cm.getLine(cm.getCursor().line + i)); });
|
||||||
|
},
|
||||||
|
"/": function(cm) {var f = CodeMirror.commands.find; f && f(cm); sdir = "f";},
|
||||||
|
"'?'": function(cm) {
|
||||||
|
var f = CodeMirror.commands.find;
|
||||||
|
if (f) { f(cm); CodeMirror.commands.findPrev(cm); sdir = "r"; }
|
||||||
|
},
|
||||||
|
"N": function(cm) {
|
||||||
|
var fn = CodeMirror.commands.findNext;
|
||||||
|
if (fn) sdir != "r" ? fn(cm) : CodeMirror.commands.findPrev(cm);
|
||||||
|
},
|
||||||
|
"Shift-N": function(cm) {
|
||||||
|
var fn = CodeMirror.commands.findNext;
|
||||||
|
if (fn) sdir != "r" ? CodeMirror.commands.findPrev(cm) : fn.findNext(cm);
|
||||||
|
},
|
||||||
|
"Shift-G": function(cm) {
|
||||||
|
(repeatCount == 0) ? cm.setCursor(cm.lineCount()) : cm.setCursor(repeatCount - 1);
|
||||||
|
clearCount();
|
||||||
|
CodeMirror.commands.goLineStart(cm);
|
||||||
|
},
|
||||||
|
"':'": function(cm) {
|
||||||
|
var exModeDialog = ': <input type="text" style="width: 90%"/>';
|
||||||
|
dialog(cm, exModeDialog, ':', function(command) {
|
||||||
|
if (command.match(/^\d+$/)) {
|
||||||
|
cm.setCursor(command - 1, cm.getCursor().ch);
|
||||||
|
} else {
|
||||||
|
showAlert(cm, "Bad command: " + command);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
// standard mode switching
|
||||||
|
iterList(["d", "t", "T", "f", "F", "c", "r"], function (ch) {
|
||||||
|
CodeMirror.keyMap.vim[toCombo(ch)] = function (cm) {
|
||||||
|
cm.setOption("keyMap", "vim-prefix-" + ch);
|
||||||
|
emptyBuffer();
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
// main num keymap
|
||||||
|
// Add bindings that are influenced by number keys
|
||||||
|
iterObj({
|
||||||
|
"X": function(cm) {CodeMirror.commands.delCharRight(cm);},
|
||||||
|
"P": function(cm) {
|
||||||
|
var cur = cm.getCursor().line;
|
||||||
|
if (buf!= "") {
|
||||||
|
if (buf[0] == "\n") CodeMirror.commands.goLineEnd(cm);
|
||||||
|
cm.replaceRange(buf, cm.getCursor());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Shift-X": function(cm) {CodeMirror.commands.delCharLeft(cm);},
|
||||||
|
"Shift-J": function(cm) {joinLineNext(cm);},
|
||||||
|
"Shift-P": function(cm) {
|
||||||
|
var cur = cm.getCursor().line;
|
||||||
|
if (buf!= "") {
|
||||||
|
CodeMirror.commands.goLineUp(cm);
|
||||||
|
CodeMirror.commands.goLineEnd(cm);
|
||||||
|
cm.replaceSelection(buf, "end");
|
||||||
|
}
|
||||||
|
cm.setCursor(cur+1);
|
||||||
|
},
|
||||||
|
"'~'": function(cm) {
|
||||||
|
var cur = cm.getCursor(), cHar = cm.getRange({line: cur.line, ch: cur.ch}, {line: cur.line, ch: cur.ch+1});
|
||||||
|
cHar = cHar != cHar.toLowerCase() ? cHar.toLowerCase() : cHar.toUpperCase();
|
||||||
|
cm.replaceRange(cHar, {line: cur.line, ch: cur.ch}, {line: cur.line, ch: cur.ch+1});
|
||||||
|
cm.setCursor(cur.line, cur.ch+1);
|
||||||
|
},
|
||||||
|
"Ctrl-B": function(cm) {CodeMirror.commands.goPageUp(cm);},
|
||||||
|
"Ctrl-F": function(cm) {CodeMirror.commands.goPageDown(cm);},
|
||||||
|
"Ctrl-P": "goLineUp", "Ctrl-N": "goLineDown",
|
||||||
|
"U": "undo", "Ctrl-R": "redo"
|
||||||
|
}, function(key, cmd) { map[key] = countTimes(cmd); });
|
||||||
|
|
||||||
|
// empty key maps
|
||||||
|
iterList([
|
||||||
|
"vim-prefix-d'",
|
||||||
|
"vim-prefix-y'",
|
||||||
|
"vim-prefix-df",
|
||||||
|
"vim-prefix-dF",
|
||||||
|
"vim-prefix-dt",
|
||||||
|
"vim-prefix-dT",
|
||||||
|
"vim-prefix-c",
|
||||||
|
"vim-prefix-cf",
|
||||||
|
"vim-prefix-cF",
|
||||||
|
"vim-prefix-ct",
|
||||||
|
"vim-prefix-cT",
|
||||||
|
"vim-prefix-",
|
||||||
|
"vim-prefix-f",
|
||||||
|
"vim-prefix-F",
|
||||||
|
"vim-prefix-t",
|
||||||
|
"vim-prefix-T",
|
||||||
|
"vim-prefix-r",
|
||||||
|
"vim-prefix-m"
|
||||||
|
],
|
||||||
|
function (prefix) {
|
||||||
|
CodeMirror.keyMap[prefix] = {
|
||||||
|
auto: "vim",
|
||||||
|
nofallthrough: true,
|
||||||
|
style: "fat-cursor"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.keyMap["vim-prefix-g"] = {
|
||||||
|
"E": countTimes(function(cm) { cm.setCursor(moveToWord(cm, word, -1, 1, "end"));}),
|
||||||
|
"Shift-E": countTimes(function(cm) { cm.setCursor(moveToWord(cm, bigWord, -1, 1, "end"));}),
|
||||||
|
"G": function (cm) {
|
||||||
|
cm.setCursor({line: repeatCount - 1, ch: cm.getCursor().ch});
|
||||||
|
clearCount();
|
||||||
|
},
|
||||||
|
auto: "vim", nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap["vim-prefix-d"] = {
|
||||||
|
"D": countTimes(function(cm) {
|
||||||
|
pushInBuffer("\n" + cm.getLine(cm.getCursor().line));
|
||||||
|
cm.removeLine(cm.getCursor().line);
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
}),
|
||||||
|
"'": function(cm) {
|
||||||
|
cm.setOption("keyMap", "vim-prefix-d'");
|
||||||
|
emptyBuffer();
|
||||||
|
},
|
||||||
|
"B": function(cm) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
var line = cm.getLine(cur.line);
|
||||||
|
var index = line.lastIndexOf(" ", cur.ch);
|
||||||
|
|
||||||
|
pushInBuffer(line.substring(index, cur.ch));
|
||||||
|
cm.replaceRange("", {line: cur.line, ch: index}, cur);
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
},
|
||||||
|
nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap["vim-prefix-c"] = {
|
||||||
|
"B": function (cm) {
|
||||||
|
countTimes("delWordLeft")(cm);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
"C": function (cm) {
|
||||||
|
iterTimes(function (i, last) {
|
||||||
|
CodeMirror.commands.deleteLine(cm);
|
||||||
|
if (i) {
|
||||||
|
CodeMirror.commands.delCharRight(cm);
|
||||||
|
if (last) CodeMirror.commands.deleteLine(cm);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
enterInsertMode(cm);
|
||||||
|
},
|
||||||
|
nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
iterList(["vim-prefix-d", "vim-prefix-c", "vim-prefix-"], function (prefix) {
|
||||||
|
iterList(["f", "F", "T", "t"],
|
||||||
|
function (ch) {
|
||||||
|
CodeMirror.keyMap[prefix][toCombo(ch)] = function (cm) {
|
||||||
|
cm.setOption("keyMap", prefix + ch);
|
||||||
|
emptyBuffer();
|
||||||
|
};
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
var MOTION_OPTIONS = {
|
||||||
|
"t": {inclusive: false, forward: true},
|
||||||
|
"f": {inclusive: true, forward: true},
|
||||||
|
"T": {inclusive: false, forward: false},
|
||||||
|
"F": {inclusive: true, forward: false}
|
||||||
|
};
|
||||||
|
|
||||||
|
function setupPrefixBindingForKey(m) {
|
||||||
|
CodeMirror.keyMap["vim-prefix-m"][m] = function(cm) {
|
||||||
|
mark[m] = cm.getCursor().line;
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap["vim-prefix-d'"][m] = function(cm) {
|
||||||
|
delTillMark(cm, m);
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap["vim-prefix-y'"][m] = function(cm) {
|
||||||
|
yankTillMark(cm, m);
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap["vim-prefix-r"][m] = function (cm) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
cm.replaceRange(toLetter(m),
|
||||||
|
{line: cur.line, ch: cur.ch},
|
||||||
|
{line: cur.line, ch: cur.ch + 1});
|
||||||
|
CodeMirror.commands.goColumnLeft(cm);
|
||||||
|
};
|
||||||
|
// all commands, related to motions till char in line
|
||||||
|
iterObj(MOTION_OPTIONS, function (ch, options) {
|
||||||
|
CodeMirror.keyMap["vim-prefix-" + ch][m] = function(cm) {
|
||||||
|
moveTillChar(cm, m, options);
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap["vim-prefix-d" + ch][m] = function(cm) {
|
||||||
|
delTillChar(cm, m, options);
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap["vim-prefix-c" + ch][m] = function(cm) {
|
||||||
|
delTillChar(cm, m, options);
|
||||||
|
enterInsertMode(cm);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
for (var i = 65; i < 65 + 26; i++) { // uppercase alphabet char codes
|
||||||
|
var ch = String.fromCharCode(i);
|
||||||
|
setupPrefixBindingForKey(toCombo(ch));
|
||||||
|
setupPrefixBindingForKey(toCombo(ch.toLowerCase()));
|
||||||
|
}
|
||||||
|
for (var i = 0; i < SPECIAL_SYMBOLS.length; ++i) {
|
||||||
|
setupPrefixBindingForKey(toCombo(SPECIAL_SYMBOLS.charAt(i)));
|
||||||
|
}
|
||||||
|
setupPrefixBindingForKey("Space");
|
||||||
|
|
||||||
|
CodeMirror.keyMap["vim-prefix-y"] = {
|
||||||
|
"Y": countTimes(function(cm, i, last) {
|
||||||
|
pushInBuffer("\n" + cm.getLine(cm.getCursor().line + i));
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
}),
|
||||||
|
"'": function(cm) {cm.setOption("keyMap", "vim-prefix-y'"); emptyBuffer();},
|
||||||
|
nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap["vim-insert"] = {
|
||||||
|
// TODO: override navigation keys so that Esc will cancel automatic indentation from o, O, i_<CR>
|
||||||
|
"Esc": function(cm) {
|
||||||
|
cm.setCursor(cm.getCursor().line, cm.getCursor().ch-1, true);
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
},
|
||||||
|
"Ctrl-N": "autocomplete",
|
||||||
|
"Ctrl-P": "autocomplete",
|
||||||
|
fallthrough: ["default"]
|
||||||
|
};
|
||||||
|
|
||||||
|
function findMatchedSymbol(cm, cur, symb) {
|
||||||
|
var line = cur.line;
|
||||||
|
var symb = symb ? symb : cm.getLine(line)[cur.ch];
|
||||||
|
|
||||||
|
// Are we at the opening or closing char
|
||||||
|
var forwards = ['(', '[', '{'].indexOf(symb) != -1;
|
||||||
|
|
||||||
|
var reverseSymb = (function(sym) {
|
||||||
|
switch (sym) {
|
||||||
|
case '(' : return ')';
|
||||||
|
case '[' : return ']';
|
||||||
|
case '{' : return '}';
|
||||||
|
case ')' : return '(';
|
||||||
|
case ']' : return '[';
|
||||||
|
case '}' : return '{';
|
||||||
|
default : return null;
|
||||||
|
}
|
||||||
|
})(symb);
|
||||||
|
|
||||||
|
// Couldn't find a matching symbol, abort
|
||||||
|
if (reverseSymb == null) return cur;
|
||||||
|
|
||||||
|
// Tracking our imbalance in open/closing symbols. An opening symbol wii be
|
||||||
|
// the first thing we pick up if moving forward, this isn't true moving backwards
|
||||||
|
var disBal = forwards ? 0 : 1;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (line == cur.line) {
|
||||||
|
// First pass, do some special stuff
|
||||||
|
var currLine = forwards ? cm.getLine(line).substr(cur.ch).split('') : cm.getLine(line).substr(0,cur.ch).split('').reverse();
|
||||||
|
} else {
|
||||||
|
var currLine = forwards ? cm.getLine(line).split('') : cm.getLine(line).split('').reverse();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var index = 0; index < currLine.length; index++) {
|
||||||
|
if (currLine[index] == symb) disBal++;
|
||||||
|
else if (currLine[index] == reverseSymb) disBal--;
|
||||||
|
|
||||||
|
if (disBal == 0) {
|
||||||
|
if (forwards && cur.line == line) return {line: line, ch: index + cur.ch};
|
||||||
|
else if (forwards) return {line: line, ch: index};
|
||||||
|
else return {line: line, ch: currLine.length - index - 1 };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (forwards) line++;
|
||||||
|
else line--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectCompanionObject(cm, revSymb, inclusive) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
|
||||||
|
var end = findMatchedSymbol(cm, cur, revSymb);
|
||||||
|
var start = findMatchedSymbol(cm, end);
|
||||||
|
start.ch += inclusive ? 1 : 0;
|
||||||
|
end.ch += inclusive ? 0 : 1;
|
||||||
|
|
||||||
|
return {start: start, end: end};
|
||||||
|
}
|
||||||
|
|
||||||
|
// takes in a symbol and a cursor and tries to simulate text objects that have
|
||||||
|
// identical opening and closing symbols
|
||||||
|
// TODO support across multiple lines
|
||||||
|
function findBeginningAndEnd(cm, symb, inclusive) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
var line = cm.getLine(cur.line);
|
||||||
|
var chars = line.split('');
|
||||||
|
var start = undefined;
|
||||||
|
var end = undefined;
|
||||||
|
var firstIndex = chars.indexOf(symb);
|
||||||
|
|
||||||
|
// the decision tree is to always look backwards for the beginning first,
|
||||||
|
// but if the cursor is in front of the first instance of the symb,
|
||||||
|
// then move the cursor forward
|
||||||
|
if (cur.ch < firstIndex) {
|
||||||
|
cur.ch = firstIndex;
|
||||||
|
cm.setCursor(cur.line, firstIndex+1);
|
||||||
|
}
|
||||||
|
// otherwise if the cursor is currently on the closing symbol
|
||||||
|
else if (firstIndex < cur.ch && chars[cur.ch] == symb) {
|
||||||
|
end = cur.ch; // assign end to the current cursor
|
||||||
|
--cur.ch; // make sure to look backwards
|
||||||
|
}
|
||||||
|
|
||||||
|
// if we're currently on the symbol, we've got a start
|
||||||
|
if (chars[cur.ch] == symb && end == null)
|
||||||
|
start = cur.ch + 1; // assign start to ahead of the cursor
|
||||||
|
else {
|
||||||
|
// go backwards to find the start
|
||||||
|
for (var i = cur.ch; i > -1 && start == null; i--)
|
||||||
|
if (chars[i] == symb) start = i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// look forwards for the end symbol
|
||||||
|
if (start != null && end == null) {
|
||||||
|
for (var i = start, len = chars.length; i < len && end == null; i++) {
|
||||||
|
if (chars[i] == symb) end = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// nothing found
|
||||||
|
// FIXME still enters insert mode
|
||||||
|
if (start == null || end == null) return {
|
||||||
|
start: cur, end: cur
|
||||||
|
};
|
||||||
|
|
||||||
|
// include the symbols
|
||||||
|
if (inclusive) {
|
||||||
|
--start; ++end;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
start: {line: cur.line, ch: start},
|
||||||
|
end: {line: cur.line, ch: end}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function offsetCursor(cm, line, ch) {
|
||||||
|
var cur = cm.getCursor(); return {line: cur.line + line, ch: cur.ch + ch};
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are the motion commands we use for navigation and selection with
|
||||||
|
// certain other commands. All should return a cursor object.
|
||||||
|
var motions = {
|
||||||
|
"J": function(cm, times) { return offsetCursor(cm, times, 0); },
|
||||||
|
"Down": function(cm, times) { return offsetCursor(cm, times, 0); },
|
||||||
|
"K": function(cm, times) { return offsetCursor(cm, -times, 0); },
|
||||||
|
"Up": function(cm, times) { return offsetCursor(cm, -times, 0); },
|
||||||
|
"L": function(cm, times) { return offsetCursor(cm, 0, times); },
|
||||||
|
"Right": function(cm, times) { return offsetCursor(cm, 0, times); },
|
||||||
|
"Space": function(cm, times) { return offsetCursor(cm, 0, times); },
|
||||||
|
"H": function(cm, times) { return offsetCursor(cm, 0, -times); },
|
||||||
|
"Left": function(cm, times) { return offsetCursor(cm, 0, -times); },
|
||||||
|
"Backspace": function(cm, times) { return offsetCursor(cm, 0, -times); },
|
||||||
|
"B": function(cm, times, yank) { return moveToWord(cm, word, -1, times, 'start', yank); },
|
||||||
|
"Shift-B": function(cm, times, yank) { return moveToWord(cm, bigWord, -1, times, 'start', yank); },
|
||||||
|
"E": function(cm, times, yank) { return moveToWord(cm, word, 1, times, 'end', yank); },
|
||||||
|
"Shift-E": function(cm, times, yank) { return moveToWord(cm, bigWord, 1, times, 'end', yank); },
|
||||||
|
"W": function(cm, times, yank) { return moveToWord(cm, word, 1, times, 'start', yank); },
|
||||||
|
"Shift-W": function(cm, times, yank) { return moveToWord(cm, bigWord, 1, times, 'start', yank); },
|
||||||
|
"'^'": function(cm, times) {
|
||||||
|
var cur = cm.getCursor(), line = cm.getLine(cur.line).split('');
|
||||||
|
for (var i = 0; i < line.length; i++) {
|
||||||
|
if (line[i].match(/[^\s]/)) return {line: cur.line, ch: index};
|
||||||
|
}
|
||||||
|
return cur;
|
||||||
|
},
|
||||||
|
"'$'": function(cm) {
|
||||||
|
var cur = cm.getCursor(), ch = cm.getLine(cur.line).length;
|
||||||
|
return {line: cur.line, ch: ch};
|
||||||
|
},
|
||||||
|
"'%'": function(cm) { return findMatchedSymbol(cm, cm.getCursor()); },
|
||||||
|
"Esc" : function(cm) { cm.setOption("keyMap", "vim"); repeatCount = 0; return cm.getCursor(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
// Map our movement actions each operator and non-operational movement
|
||||||
|
iterObj(motions, function(key, motion) {
|
||||||
|
CodeMirror.keyMap['vim-prefix-d'][key] = function(cm) {
|
||||||
|
// Get our selected range
|
||||||
|
var start = cm.getCursor();
|
||||||
|
var end = motion(cm, repeatCount ? repeatCount : 1, true);
|
||||||
|
|
||||||
|
// Set swap var if range is of negative length
|
||||||
|
if ((start.line > end.line) || (start.line == end.line && start.ch > end.ch)) var swap = true;
|
||||||
|
|
||||||
|
// Take action, switching start and end if swap var is set
|
||||||
|
pushInBuffer(cm.getRange(swap ? end : start, swap ? start : end));
|
||||||
|
cm.replaceRange("", swap ? end : start, swap ? start : end);
|
||||||
|
|
||||||
|
// And clean up
|
||||||
|
repeatCount = 0;
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap['vim-prefix-c'][key] = function(cm) {
|
||||||
|
var start = cm.getCursor();
|
||||||
|
var end = motion(cm, repeatCount ? repeatCount : 1, true);
|
||||||
|
|
||||||
|
if ((start.line > end.line) || (start.line == end.line && start.ch > end.ch)) var swap = true;
|
||||||
|
pushInBuffer(cm.getRange(swap ? end : start, swap ? start : end));
|
||||||
|
cm.replaceRange("", swap ? end : start, swap ? start : end);
|
||||||
|
|
||||||
|
repeatCount = 0;
|
||||||
|
cm.setOption('keyMap', 'vim-insert');
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap['vim-prefix-y'][key] = function(cm) {
|
||||||
|
var start = cm.getCursor();
|
||||||
|
var end = motion(cm, repeatCount ? repeatCount : 1, true);
|
||||||
|
|
||||||
|
if ((start.line > end.line) || (start.line == end.line && start.ch > end.ch)) var swap = true;
|
||||||
|
pushInBuffer(cm.getRange(swap ? end : start, swap ? start : end));
|
||||||
|
|
||||||
|
repeatCount = 0;
|
||||||
|
cm.setOption("keyMap", "vim");
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap['vim'][key] = function(cm) {
|
||||||
|
var cur = motion(cm, repeatCount ? repeatCount : 1);
|
||||||
|
cm.setCursor(cur.line, cur.ch);
|
||||||
|
|
||||||
|
repeatCount = 0;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
function addCountBindings(keyMapName) {
|
||||||
|
// Add bindings for number keys
|
||||||
|
keyMap = CodeMirror.keyMap[keyMapName];
|
||||||
|
keyMap["0"] = function(cm) {
|
||||||
|
if (repeatCount > 0) {
|
||||||
|
pushRepeatCountDigit(0)(cm);
|
||||||
|
} else {
|
||||||
|
CodeMirror.commands.goLineStart(cm);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
for (var i = 1; i < 10; ++i) {
|
||||||
|
keyMap[i] = pushRepeatCountDigit(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
addCountBindings('vim');
|
||||||
|
addCountBindings('vim-prefix-d');
|
||||||
|
addCountBindings('vim-prefix-y');
|
||||||
|
addCountBindings('vim-prefix-c');
|
||||||
|
|
||||||
|
// Create our keymaps for each operator and make xa and xi where x is an operator
|
||||||
|
// change to the corrosponding keymap
|
||||||
|
var operators = ['d', 'y', 'c'];
|
||||||
|
iterList(operators, function(key, index, array) {
|
||||||
|
CodeMirror.keyMap['vim-prefix-'+key+'a'] = {
|
||||||
|
auto: 'vim', nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
CodeMirror.keyMap['vim-prefix-'+key+'i'] = {
|
||||||
|
auto: 'vim', nofallthrough: true, style: "fat-cursor"
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap['vim-prefix-'+key]['A'] = function(cm) {
|
||||||
|
repeatCount = 0;
|
||||||
|
cm.setOption('keyMap', 'vim-prefix-' + key + 'a');
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.keyMap['vim-prefix-'+key]['I'] = function(cm) {
|
||||||
|
repeatCount = 0;
|
||||||
|
cm.setOption('keyMap', 'vim-prefix-' + key + 'i');
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
function regexLastIndexOf(string, pattern, startIndex) {
|
||||||
|
for (var i = startIndex == null ? string.length : startIndex; i >= 0; --i)
|
||||||
|
if (pattern.test(string.charAt(i))) return i;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create our text object functions. They work similar to motions but they
|
||||||
|
// return a start cursor as well
|
||||||
|
var textObjectList = ['W', 'Shift-[', 'Shift-9', '[', "'", "Shift-'"];
|
||||||
|
var textObjects = {
|
||||||
|
'W': function(cm, inclusive) {
|
||||||
|
var cur = cm.getCursor();
|
||||||
|
var line = cm.getLine(cur.line);
|
||||||
|
|
||||||
|
var line_to_char = new String(line.substring(0, cur.ch));
|
||||||
|
var start = regexLastIndexOf(line_to_char, /[^a-zA-Z0-9]/) + 1;
|
||||||
|
var end = motions["E"](cm, 1) ;
|
||||||
|
|
||||||
|
end.ch += inclusive ? 1 : 0 ;
|
||||||
|
return {start: {line: cur.line, ch: start}, end: end };
|
||||||
|
},
|
||||||
|
'Shift-[': function(cm, inclusive) { return selectCompanionObject(cm, '}', inclusive); },
|
||||||
|
'Shift-9': function(cm, inclusive) { return selectCompanionObject(cm, ')', inclusive); },
|
||||||
|
'[': function(cm, inclusive) { return selectCompanionObject(cm, ']', inclusive); },
|
||||||
|
"'": function(cm, inclusive) { return findBeginningAndEnd(cm, "'", inclusive); },
|
||||||
|
"Shift-'": function(cm, inclusive) { return findBeginningAndEnd(cm, '"', inclusive); }
|
||||||
|
};
|
||||||
|
|
||||||
|
// One function to handle all operation upon text objects. Kinda funky but it works
|
||||||
|
// better than rewriting this code six times
|
||||||
|
function textObjectManipulation(cm, object, remove, insert, inclusive) {
|
||||||
|
// Object is the text object, delete object if remove is true, enter insert
|
||||||
|
// mode if insert is true, inclusive is the difference between a and i
|
||||||
|
var tmp = textObjects[object](cm, inclusive);
|
||||||
|
var start = tmp.start;
|
||||||
|
var end = tmp.end;
|
||||||
|
|
||||||
|
if ((start.line > end.line) || (start.line == end.line && start.ch > end.ch)) var swap = true ;
|
||||||
|
|
||||||
|
pushInBuffer(cm.getRange(swap ? end : start, swap ? start : end));
|
||||||
|
if (remove) cm.replaceRange("", swap ? end : start, swap ? start : end);
|
||||||
|
if (insert) cm.setOption('keyMap', 'vim-insert');
|
||||||
|
}
|
||||||
|
|
||||||
|
// And finally build the keymaps up from the text objects
|
||||||
|
for (var i = 0; i < textObjectList.length; ++i) {
|
||||||
|
var object = textObjectList[i];
|
||||||
|
(function(object) {
|
||||||
|
CodeMirror.keyMap['vim-prefix-di'][object] = function(cm) { textObjectManipulation(cm, object, true, false, false); };
|
||||||
|
CodeMirror.keyMap['vim-prefix-da'][object] = function(cm) { textObjectManipulation(cm, object, true, false, true); };
|
||||||
|
CodeMirror.keyMap['vim-prefix-yi'][object] = function(cm) { textObjectManipulation(cm, object, false, false, false); };
|
||||||
|
CodeMirror.keyMap['vim-prefix-ya'][object] = function(cm) { textObjectManipulation(cm, object, false, false, true); };
|
||||||
|
CodeMirror.keyMap['vim-prefix-ci'][object] = function(cm) { textObjectManipulation(cm, object, true, true, false); };
|
||||||
|
CodeMirror.keyMap['vim-prefix-ca'][object] = function(cm) { textObjectManipulation(cm, object, true, true, true); };
|
||||||
|
})(object)
|
||||||
|
}
|
||||||
|
})();
|
174
codemirror-2.36/lib/codemirror.css
Normal file
174
codemirror-2.36/lib/codemirror.css
Normal file
|
@ -0,0 +1,174 @@
|
||||||
|
.CodeMirror {
|
||||||
|
line-height: 1em;
|
||||||
|
font-family: monospace;
|
||||||
|
|
||||||
|
/* Necessary so the scrollbar can be absolutely positioned within the wrapper on Lion. */
|
||||||
|
position: relative;
|
||||||
|
/* This prevents unwanted scrollbars from showing up on the body and wrapper in IE. */
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-scroll {
|
||||||
|
overflow: auto;
|
||||||
|
height: 300px;
|
||||||
|
/* This is needed to prevent an IE[67] bug where the scrolled content
|
||||||
|
is visible outside of the scrolling box. */
|
||||||
|
position: relative;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Vertical scrollbar */
|
||||||
|
.CodeMirror-scrollbar {
|
||||||
|
position: absolute;
|
||||||
|
right: 0; top: 0;
|
||||||
|
overflow-x: hidden;
|
||||||
|
overflow-y: scroll;
|
||||||
|
z-index: 5;
|
||||||
|
}
|
||||||
|
.CodeMirror-scrollbar-inner {
|
||||||
|
/* This needs to have a nonzero width in order for the scrollbar to appear
|
||||||
|
in Firefox and IE9. */
|
||||||
|
width: 1px;
|
||||||
|
}
|
||||||
|
.CodeMirror-scrollbar.cm-sb-overlap {
|
||||||
|
/* Ensure that the scrollbar appears in Lion, and that it overlaps the content
|
||||||
|
rather than sitting to the right of it. */
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
float: none;
|
||||||
|
right: 0;
|
||||||
|
min-width: 12px;
|
||||||
|
}
|
||||||
|
.CodeMirror-scrollbar.cm-sb-nonoverlap {
|
||||||
|
min-width: 12px;
|
||||||
|
}
|
||||||
|
.CodeMirror-scrollbar.cm-sb-ie7 {
|
||||||
|
min-width: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-gutter {
|
||||||
|
position: absolute; left: 0; top: 0;
|
||||||
|
z-index: 10;
|
||||||
|
background-color: #f7f7f7;
|
||||||
|
border-right: 1px solid #eee;
|
||||||
|
min-width: 2em;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.CodeMirror-gutter-text {
|
||||||
|
color: #aaa;
|
||||||
|
text-align: right;
|
||||||
|
padding: .4em .2em .4em .4em;
|
||||||
|
white-space: pre !important;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
.CodeMirror-lines {
|
||||||
|
padding: .4em;
|
||||||
|
white-space: pre;
|
||||||
|
cursor: text;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror pre {
|
||||||
|
-moz-border-radius: 0;
|
||||||
|
-webkit-border-radius: 0;
|
||||||
|
-o-border-radius: 0;
|
||||||
|
border-radius: 0;
|
||||||
|
border-width: 0; margin: 0; padding: 0; background: transparent;
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: inherit;
|
||||||
|
padding: 0; margin: 0;
|
||||||
|
white-space: pre;
|
||||||
|
word-wrap: normal;
|
||||||
|
line-height: inherit;
|
||||||
|
color: inherit;
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-wrap pre {
|
||||||
|
word-wrap: break-word;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: normal;
|
||||||
|
}
|
||||||
|
.CodeMirror-wrap .CodeMirror-scroll {
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror textarea {
|
||||||
|
outline: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror pre.CodeMirror-cursor {
|
||||||
|
z-index: 10;
|
||||||
|
position: absolute;
|
||||||
|
visibility: hidden;
|
||||||
|
border-left: 1px solid black;
|
||||||
|
border-right: none;
|
||||||
|
width: 0;
|
||||||
|
}
|
||||||
|
.cm-keymap-fat-cursor pre.CodeMirror-cursor {
|
||||||
|
width: auto;
|
||||||
|
border: 0;
|
||||||
|
background: transparent;
|
||||||
|
background: rgba(0, 200, 0, .4);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr=#6600c800, endColorstr=#4c00c800);
|
||||||
|
}
|
||||||
|
/* Kludge to turn off filter in ie9+, which also accepts rgba */
|
||||||
|
.cm-keymap-fat-cursor pre.CodeMirror-cursor:not(#nonsense_id) {
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
}
|
||||||
|
.CodeMirror pre.CodeMirror-cursor.CodeMirror-overwrite {}
|
||||||
|
.CodeMirror-focused pre.CodeMirror-cursor {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.CodeMirror-selected { background: #d9d9d9; }
|
||||||
|
.CodeMirror-focused div.CodeMirror-selected { background: #d7d4f0; }
|
||||||
|
|
||||||
|
.CodeMirror-searching {
|
||||||
|
background: #ffa;
|
||||||
|
background: rgba(255, 255, 0, .4);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Default theme */
|
||||||
|
|
||||||
|
.cm-s-default span.cm-keyword {color: #708;}
|
||||||
|
.cm-s-default span.cm-atom {color: #219;}
|
||||||
|
.cm-s-default span.cm-number {color: #164;}
|
||||||
|
.cm-s-default span.cm-def {color: #00f;}
|
||||||
|
.cm-s-default span.cm-variable {color: black;}
|
||||||
|
.cm-s-default span.cm-variable-2 {color: #05a;}
|
||||||
|
.cm-s-default span.cm-variable-3 {color: #085;}
|
||||||
|
.cm-s-default span.cm-property {color: black;}
|
||||||
|
.cm-s-default span.cm-operator {color: black;}
|
||||||
|
.cm-s-default span.cm-comment {color: #a50;}
|
||||||
|
.cm-s-default span.cm-string {color: #a11;}
|
||||||
|
.cm-s-default span.cm-string-2 {color: #f50;}
|
||||||
|
.cm-s-default span.cm-meta {color: #555;}
|
||||||
|
.cm-s-default span.cm-error {color: #f00;}
|
||||||
|
.cm-s-default span.cm-qualifier {color: #555;}
|
||||||
|
.cm-s-default span.cm-builtin {color: #30a;}
|
||||||
|
.cm-s-default span.cm-bracket {color: #997;}
|
||||||
|
.cm-s-default span.cm-tag {color: #170;}
|
||||||
|
.cm-s-default span.cm-attribute {color: #00c;}
|
||||||
|
.cm-s-default span.cm-header {color: blue;}
|
||||||
|
.cm-s-default span.cm-quote {color: #090;}
|
||||||
|
.cm-s-default span.cm-hr {color: #999;}
|
||||||
|
.cm-s-default span.cm-link {color: #00c;}
|
||||||
|
|
||||||
|
span.cm-header, span.cm-strong {font-weight: bold;}
|
||||||
|
span.cm-em {font-style: italic;}
|
||||||
|
span.cm-emstrong {font-style: italic; font-weight: bold;}
|
||||||
|
span.cm-link {text-decoration: underline;}
|
||||||
|
|
||||||
|
span.cm-invalidchar {color: #f00;}
|
||||||
|
|
||||||
|
div.CodeMirror span.CodeMirror-matchingbracket {color: #0f0;}
|
||||||
|
div.CodeMirror span.CodeMirror-nonmatchingbracket {color: #f22;}
|
||||||
|
|
||||||
|
@media print {
|
||||||
|
|
||||||
|
/* Hide the cursor when printing */
|
||||||
|
.CodeMirror pre.CodeMirror-cursor {
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
3171
codemirror-2.36/lib/codemirror.js
Normal file
3171
codemirror-2.36/lib/codemirror.js
Normal file
File diff suppressed because it is too large
Load Diff
164
codemirror-2.36/lib/util/closetag.js
Normal file
164
codemirror-2.36/lib/util/closetag.js
Normal file
|
@ -0,0 +1,164 @@
|
||||||
|
/**
|
||||||
|
* Tag-closer extension for CodeMirror.
|
||||||
|
*
|
||||||
|
* This extension adds a "closeTag" utility function that can be used with key bindings to
|
||||||
|
* insert a matching end tag after the ">" character of a start tag has been typed. It can
|
||||||
|
* also complete "</" if a matching start tag is found. It will correctly ignore signal
|
||||||
|
* characters for empty tags, comments, CDATA, etc.
|
||||||
|
*
|
||||||
|
* The function depends on internal parser state to identify tags. It is compatible with the
|
||||||
|
* following CodeMirror modes and will ignore all others:
|
||||||
|
* - htmlmixed
|
||||||
|
* - xml
|
||||||
|
*
|
||||||
|
* See demos/closetag.html for a usage example.
|
||||||
|
*
|
||||||
|
* @author Nathan Williams <nathan@nlwillia.net>
|
||||||
|
* Contributed under the same license terms as CodeMirror.
|
||||||
|
*/
|
||||||
|
(function() {
|
||||||
|
/** Option that allows tag closing behavior to be toggled. Default is true. */
|
||||||
|
CodeMirror.defaults['closeTagEnabled'] = true;
|
||||||
|
|
||||||
|
/** Array of tag names to add indentation after the start tag for. Default is the list of block-level html tags. */
|
||||||
|
CodeMirror.defaults['closeTagIndent'] = ['applet', 'blockquote', 'body', 'button', 'div', 'dl', 'fieldset', 'form', 'frameset', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'head', 'html', 'iframe', 'layer', 'legend', 'object', 'ol', 'p', 'select', 'table', 'ul'];
|
||||||
|
|
||||||
|
/** Array of tag names where an end tag is forbidden. */
|
||||||
|
CodeMirror.defaults['closeTagVoid'] = ['area', 'base', 'br', 'col', 'command', 'embed', 'hr', 'img', 'input', 'keygen', 'link', 'meta', 'param', 'source', 'track', 'wbr'];
|
||||||
|
|
||||||
|
function innerState(cm, state) {
|
||||||
|
return CodeMirror.innerMode(cm.getMode(), state).state;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call during key processing to close tags. Handles the key event if the tag is closed, otherwise throws CodeMirror.Pass.
|
||||||
|
* - cm: The editor instance.
|
||||||
|
* - ch: The character being processed.
|
||||||
|
* - indent: Optional. An array of tag names to indent when closing. Omit or pass true to use the default indentation tag list defined in the 'closeTagIndent' option.
|
||||||
|
* Pass false to disable indentation. Pass an array to override the default list of tag names.
|
||||||
|
* - vd: Optional. An array of tag names that should not be closed. Omit to use the default void (end tag forbidden) tag list defined in the 'closeTagVoid' option. Ignored in xml mode.
|
||||||
|
*/
|
||||||
|
CodeMirror.defineExtension("closeTag", function(cm, ch, indent, vd) {
|
||||||
|
if (!cm.getOption('closeTagEnabled')) {
|
||||||
|
throw CodeMirror.Pass;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Relevant structure of token:
|
||||||
|
*
|
||||||
|
* htmlmixed
|
||||||
|
* className
|
||||||
|
* state
|
||||||
|
* htmlState
|
||||||
|
* type
|
||||||
|
* tagName
|
||||||
|
* context
|
||||||
|
* tagName
|
||||||
|
* mode
|
||||||
|
*
|
||||||
|
* xml
|
||||||
|
* className
|
||||||
|
* state
|
||||||
|
* tagName
|
||||||
|
* type
|
||||||
|
*/
|
||||||
|
|
||||||
|
var pos = cm.getCursor();
|
||||||
|
var tok = cm.getTokenAt(pos);
|
||||||
|
var state = innerState(cm, tok.state);
|
||||||
|
|
||||||
|
if (state) {
|
||||||
|
|
||||||
|
if (ch == '>') {
|
||||||
|
var type = state.type;
|
||||||
|
|
||||||
|
if (tok.className == 'tag' && type == 'closeTag') {
|
||||||
|
throw CodeMirror.Pass; // Don't process the '>' at the end of an end-tag.
|
||||||
|
}
|
||||||
|
|
||||||
|
cm.replaceSelection('>'); // Mode state won't update until we finish the tag.
|
||||||
|
pos = {line: pos.line, ch: pos.ch + 1};
|
||||||
|
cm.setCursor(pos);
|
||||||
|
|
||||||
|
tok = cm.getTokenAt(cm.getCursor());
|
||||||
|
state = innerState(cm, tok.state);
|
||||||
|
if (!state) throw CodeMirror.Pass;
|
||||||
|
var type = state.type;
|
||||||
|
|
||||||
|
if (tok.className == 'tag' && type != 'selfcloseTag') {
|
||||||
|
var tagName = state.tagName;
|
||||||
|
if (tagName.length > 0 && shouldClose(cm, vd, tagName)) {
|
||||||
|
insertEndTag(cm, indent, pos, tagName);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Undo the '>' insert and allow cm to handle the key instead.
|
||||||
|
cm.setSelection({line: pos.line, ch: pos.ch - 1}, pos);
|
||||||
|
cm.replaceSelection("");
|
||||||
|
|
||||||
|
} else if (ch == '/') {
|
||||||
|
if (tok.className == 'tag' && tok.string == '<') {
|
||||||
|
var ctx = state.context, tagName = ctx ? ctx.tagName : '';
|
||||||
|
if (tagName.length > 0) {
|
||||||
|
completeEndTag(cm, pos, tagName);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
throw CodeMirror.Pass; // Bubble if not handled
|
||||||
|
});
|
||||||
|
|
||||||
|
function insertEndTag(cm, indent, pos, tagName) {
|
||||||
|
if (shouldIndent(cm, indent, tagName)) {
|
||||||
|
cm.replaceSelection('\n\n</' + tagName + '>', 'end');
|
||||||
|
cm.indentLine(pos.line + 1);
|
||||||
|
cm.indentLine(pos.line + 2);
|
||||||
|
cm.setCursor({line: pos.line + 1, ch: cm.getLine(pos.line + 1).length});
|
||||||
|
} else {
|
||||||
|
cm.replaceSelection('</' + tagName + '>');
|
||||||
|
cm.setCursor(pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldIndent(cm, indent, tagName) {
|
||||||
|
if (typeof indent == 'undefined' || indent == null || indent == true) {
|
||||||
|
indent = cm.getOption('closeTagIndent');
|
||||||
|
}
|
||||||
|
if (!indent) {
|
||||||
|
indent = [];
|
||||||
|
}
|
||||||
|
return indexOf(indent, tagName.toLowerCase()) != -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldClose(cm, vd, tagName) {
|
||||||
|
if (cm.getOption('mode') == 'xml') {
|
||||||
|
return true; // always close xml tags
|
||||||
|
}
|
||||||
|
if (typeof vd == 'undefined' || vd == null) {
|
||||||
|
vd = cm.getOption('closeTagVoid');
|
||||||
|
}
|
||||||
|
if (!vd) {
|
||||||
|
vd = [];
|
||||||
|
}
|
||||||
|
return indexOf(vd, tagName.toLowerCase()) == -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// C&P from codemirror.js...would be nice if this were visible to utilities.
|
||||||
|
function indexOf(collection, elt) {
|
||||||
|
if (collection.indexOf) return collection.indexOf(elt);
|
||||||
|
for (var i = 0, e = collection.length; i < e; ++i)
|
||||||
|
if (collection[i] == elt) return i;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
function completeEndTag(cm, pos, tagName) {
|
||||||
|
cm.replaceSelection('/' + tagName + '>');
|
||||||
|
cm.setCursor({line: pos.line, ch: pos.ch + tagName.length + 2 });
|
||||||
|
}
|
||||||
|
|
||||||
|
})();
|
36
codemirror-2.36/lib/util/continuecomment.js
Normal file
36
codemirror-2.36/lib/util/continuecomment.js
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
(function() {
|
||||||
|
var modes = ["clike", "css", "javascript"];
|
||||||
|
for (var i = 0; i < modes.length; ++i)
|
||||||
|
CodeMirror.extendMode(modes[i], {blockCommentStart: "/*",
|
||||||
|
blockCommentEnd: "*/",
|
||||||
|
blockCommentContinue: " * "});
|
||||||
|
|
||||||
|
CodeMirror.commands.newlineAndIndentContinueComment = function(cm) {
|
||||||
|
var pos = cm.getCursor(), token = cm.getTokenAt(pos);
|
||||||
|
var mode = CodeMirror.innerMode(cm.getMode(), token.state).mode;
|
||||||
|
var space;
|
||||||
|
|
||||||
|
if (token.className == "comment" && mode.blockCommentStart) {
|
||||||
|
var end = token.string.indexOf(mode.blockCommentEnd);
|
||||||
|
var full = cm.getRange({line: pos.line, ch: 0}, {line: pos.line, ch: token.end}), found;
|
||||||
|
if (end != -1 && end == token.string.length - mode.blockCommentEnd.length) {
|
||||||
|
// Comment ended, don't continue it
|
||||||
|
} else if (token.string.indexOf(mode.blockCommentStart) == 0) {
|
||||||
|
space = full.slice(0, token.start);
|
||||||
|
if (!/^\s*$/.test(space)) {
|
||||||
|
space = "";
|
||||||
|
for (var i = 0; i < token.start; ++i) space += " ";
|
||||||
|
}
|
||||||
|
} else if ((found = full.indexOf(mode.blockCommentContinue)) != -1 &&
|
||||||
|
found + mode.blockCommentContinue.length > token.start &&
|
||||||
|
/^\s*$/.test(full.slice(0, found))) {
|
||||||
|
space = full.slice(0, found);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (space != null)
|
||||||
|
cm.replaceSelection("\n" + space + mode.blockCommentContinue, "end");
|
||||||
|
else
|
||||||
|
cm.execCommand("newlineAndIndent");
|
||||||
|
};
|
||||||
|
})();
|
27
codemirror-2.36/lib/util/dialog.css
Normal file
27
codemirror-2.36/lib/util/dialog.css
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
.CodeMirror-dialog {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-dialog > div {
|
||||||
|
position: absolute;
|
||||||
|
top: 0; left: 0; right: 0;
|
||||||
|
background: white;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
z-index: 15;
|
||||||
|
padding: .1em .8em;
|
||||||
|
overflow: hidden;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-dialog input {
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
background: transparent;
|
||||||
|
width: 20em;
|
||||||
|
color: inherit;
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.CodeMirror-dialog button {
|
||||||
|
font-size: 70%;
|
||||||
|
}
|
70
codemirror-2.36/lib/util/dialog.js
Normal file
70
codemirror-2.36/lib/util/dialog.js
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
// Open simple dialogs on top of an editor. Relies on dialog.css.
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
function dialogDiv(cm, template) {
|
||||||
|
var wrap = cm.getWrapperElement();
|
||||||
|
var dialog = wrap.insertBefore(document.createElement("div"), wrap.firstChild);
|
||||||
|
dialog.className = "CodeMirror-dialog";
|
||||||
|
dialog.innerHTML = '<div>' + template + '</div>';
|
||||||
|
return dialog;
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.defineExtension("openDialog", function(template, callback) {
|
||||||
|
var dialog = dialogDiv(this, template);
|
||||||
|
var closed = false, me = this;
|
||||||
|
function close() {
|
||||||
|
if (closed) return;
|
||||||
|
closed = true;
|
||||||
|
dialog.parentNode.removeChild(dialog);
|
||||||
|
}
|
||||||
|
var inp = dialog.getElementsByTagName("input")[0], button;
|
||||||
|
if (inp) {
|
||||||
|
CodeMirror.connect(inp, "keydown", function(e) {
|
||||||
|
if (e.keyCode == 13 || e.keyCode == 27) {
|
||||||
|
CodeMirror.e_stop(e);
|
||||||
|
close();
|
||||||
|
me.focus();
|
||||||
|
if (e.keyCode == 13) callback(inp.value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
inp.focus();
|
||||||
|
CodeMirror.connect(inp, "blur", close);
|
||||||
|
} else if (button = dialog.getElementsByTagName("button")[0]) {
|
||||||
|
CodeMirror.connect(button, "click", function() {
|
||||||
|
close();
|
||||||
|
me.focus();
|
||||||
|
});
|
||||||
|
button.focus();
|
||||||
|
CodeMirror.connect(button, "blur", close);
|
||||||
|
}
|
||||||
|
return close;
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineExtension("openConfirm", function(template, callbacks) {
|
||||||
|
var dialog = dialogDiv(this, template);
|
||||||
|
var buttons = dialog.getElementsByTagName("button");
|
||||||
|
var closed = false, me = this, blurring = 1;
|
||||||
|
function close() {
|
||||||
|
if (closed) return;
|
||||||
|
closed = true;
|
||||||
|
dialog.parentNode.removeChild(dialog);
|
||||||
|
me.focus();
|
||||||
|
}
|
||||||
|
buttons[0].focus();
|
||||||
|
for (var i = 0; i < buttons.length; ++i) {
|
||||||
|
var b = buttons[i];
|
||||||
|
(function(callback) {
|
||||||
|
CodeMirror.connect(b, "click", function(e) {
|
||||||
|
CodeMirror.e_preventDefault(e);
|
||||||
|
close();
|
||||||
|
if (callback) callback(me);
|
||||||
|
});
|
||||||
|
})(callbacks[i]);
|
||||||
|
CodeMirror.connect(b, "blur", function() {
|
||||||
|
--blurring;
|
||||||
|
setTimeout(function() { if (blurring <= 0) close(); }, 200);
|
||||||
|
});
|
||||||
|
CodeMirror.connect(b, "focus", function() { ++blurring; });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})();
|
196
codemirror-2.36/lib/util/foldcode.js
Normal file
196
codemirror-2.36/lib/util/foldcode.js
Normal file
|
@ -0,0 +1,196 @@
|
||||||
|
// the tagRangeFinder function is
|
||||||
|
// Copyright (C) 2011 by Daniel Glazman <daniel@glazman.org>
|
||||||
|
// released under the MIT license (../../LICENSE) like the rest of CodeMirror
|
||||||
|
CodeMirror.tagRangeFinder = function(cm, line, hideEnd) {
|
||||||
|
var nameStartChar = "A-Z_a-z\\u00C0-\\u00D6\\u00D8-\\u00F6\\u00F8-\\u02FF\\u0370-\\u037D\\u037F-\\u1FFF\\u200C-\\u200D\\u2070-\\u218F\\u2C00-\\u2FEF\\u3001-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFFD";
|
||||||
|
var nameChar = nameStartChar + "\-\:\.0-9\\u00B7\\u0300-\\u036F\\u203F-\\u2040";
|
||||||
|
var xmlNAMERegExp = new RegExp("^[" + nameStartChar + "][" + nameChar + "]*");
|
||||||
|
|
||||||
|
var lineText = cm.getLine(line);
|
||||||
|
var found = false;
|
||||||
|
var tag = null;
|
||||||
|
var pos = 0;
|
||||||
|
while (!found) {
|
||||||
|
pos = lineText.indexOf("<", pos);
|
||||||
|
if (-1 == pos) // no tag on line
|
||||||
|
return;
|
||||||
|
if (pos + 1 < lineText.length && lineText[pos + 1] == "/") { // closing tag
|
||||||
|
pos++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// ok we weem to have a start tag
|
||||||
|
if (!lineText.substr(pos + 1).match(xmlNAMERegExp)) { // not a tag name...
|
||||||
|
pos++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
var gtPos = lineText.indexOf(">", pos + 1);
|
||||||
|
if (-1 == gtPos) { // end of start tag not in line
|
||||||
|
var l = line + 1;
|
||||||
|
var foundGt = false;
|
||||||
|
var lastLine = cm.lineCount();
|
||||||
|
while (l < lastLine && !foundGt) {
|
||||||
|
var lt = cm.getLine(l);
|
||||||
|
var gt = lt.indexOf(">");
|
||||||
|
if (-1 != gt) { // found a >
|
||||||
|
foundGt = true;
|
||||||
|
var slash = lt.lastIndexOf("/", gt);
|
||||||
|
if (-1 != slash && slash < gt) {
|
||||||
|
var str = lineText.substr(slash, gt - slash + 1);
|
||||||
|
if (!str.match( /\/\s*\>/ )) { // yep, that's the end of empty tag
|
||||||
|
if (hideEnd === true) l++;
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
l++;
|
||||||
|
}
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var slashPos = lineText.lastIndexOf("/", gtPos);
|
||||||
|
if (-1 == slashPos) { // cannot be empty tag
|
||||||
|
found = true;
|
||||||
|
// don't continue
|
||||||
|
}
|
||||||
|
else { // empty tag?
|
||||||
|
// check if really empty tag
|
||||||
|
var str = lineText.substr(slashPos, gtPos - slashPos + 1);
|
||||||
|
if (!str.match( /\/\s*\>/ )) { // finally not empty
|
||||||
|
found = true;
|
||||||
|
// don't continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (found) {
|
||||||
|
var subLine = lineText.substr(pos + 1);
|
||||||
|
tag = subLine.match(xmlNAMERegExp);
|
||||||
|
if (tag) {
|
||||||
|
// we have an element name, wooohooo !
|
||||||
|
tag = tag[0];
|
||||||
|
// do we have the close tag on same line ???
|
||||||
|
if (-1 != lineText.indexOf("</" + tag + ">", pos)) // yep
|
||||||
|
{
|
||||||
|
found = false;
|
||||||
|
}
|
||||||
|
// we don't, so we have a candidate...
|
||||||
|
}
|
||||||
|
else
|
||||||
|
found = false;
|
||||||
|
}
|
||||||
|
if (!found)
|
||||||
|
pos++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found) {
|
||||||
|
var startTag = "(\\<\\/" + tag + "\\>)|(\\<" + tag + "\\>)|(\\<" + tag + "\\s)|(\\<" + tag + "$)";
|
||||||
|
var startTagRegExp = new RegExp(startTag, "g");
|
||||||
|
var endTag = "</" + tag + ">";
|
||||||
|
var depth = 1;
|
||||||
|
var l = line + 1;
|
||||||
|
var lastLine = cm.lineCount();
|
||||||
|
while (l < lastLine) {
|
||||||
|
lineText = cm.getLine(l);
|
||||||
|
var match = lineText.match(startTagRegExp);
|
||||||
|
if (match) {
|
||||||
|
for (var i = 0; i < match.length; i++) {
|
||||||
|
if (match[i] == endTag)
|
||||||
|
depth--;
|
||||||
|
else
|
||||||
|
depth++;
|
||||||
|
if (!depth) {
|
||||||
|
if (hideEnd === true) l++;
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
l++;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.braceRangeFinder = function(cm, line, hideEnd) {
|
||||||
|
var lineText = cm.getLine(line), at = lineText.length, startChar, tokenType;
|
||||||
|
for (;;) {
|
||||||
|
var found = lineText.lastIndexOf("{", at);
|
||||||
|
if (found < 0) break;
|
||||||
|
tokenType = cm.getTokenAt({line: line, ch: found}).className;
|
||||||
|
if (!/^(comment|string)/.test(tokenType)) { startChar = found; break; }
|
||||||
|
at = found - 1;
|
||||||
|
}
|
||||||
|
if (startChar == null || lineText.lastIndexOf("}") > startChar) return;
|
||||||
|
var count = 1, lastLine = cm.lineCount(), end;
|
||||||
|
outer: for (var i = line + 1; i < lastLine; ++i) {
|
||||||
|
var text = cm.getLine(i), pos = 0;
|
||||||
|
for (;;) {
|
||||||
|
var nextOpen = text.indexOf("{", pos), nextClose = text.indexOf("}", pos);
|
||||||
|
if (nextOpen < 0) nextOpen = text.length;
|
||||||
|
if (nextClose < 0) nextClose = text.length;
|
||||||
|
pos = Math.min(nextOpen, nextClose);
|
||||||
|
if (pos == text.length) break;
|
||||||
|
if (cm.getTokenAt({line: i, ch: pos + 1}).className == tokenType) {
|
||||||
|
if (pos == nextOpen) ++count;
|
||||||
|
else if (!--count) { end = i; break outer; }
|
||||||
|
}
|
||||||
|
++pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (end == null || end == line + 1) return;
|
||||||
|
if (hideEnd === true) end++;
|
||||||
|
return end;
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.indentRangeFinder = function(cm, line) {
|
||||||
|
var tabSize = cm.getOption("tabSize");
|
||||||
|
var myIndent = cm.getLineHandle(line).indentation(tabSize), last;
|
||||||
|
for (var i = line + 1, end = cm.lineCount(); i < end; ++i) {
|
||||||
|
var handle = cm.getLineHandle(i);
|
||||||
|
if (!/^\s*$/.test(handle.text)) {
|
||||||
|
if (handle.indentation(tabSize) <= myIndent) break;
|
||||||
|
last = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!last) return null;
|
||||||
|
return last + 1;
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.newFoldFunction = function(rangeFinder, markText, hideEnd) {
|
||||||
|
var folded = [];
|
||||||
|
if (markText == null) markText = '<div style="position: absolute; left: 2px; color:#600">▼</div>%N%';
|
||||||
|
|
||||||
|
function isFolded(cm, n) {
|
||||||
|
for (var i = 0; i < folded.length; ++i) {
|
||||||
|
var start = cm.lineInfo(folded[i].start);
|
||||||
|
if (!start) folded.splice(i--, 1);
|
||||||
|
else if (start.line == n) return {pos: i, region: folded[i]};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function expand(cm, region) {
|
||||||
|
cm.clearMarker(region.start);
|
||||||
|
for (var i = 0; i < region.hidden.length; ++i)
|
||||||
|
cm.showLine(region.hidden[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return function(cm, line) {
|
||||||
|
cm.operation(function() {
|
||||||
|
var known = isFolded(cm, line);
|
||||||
|
if (known) {
|
||||||
|
folded.splice(known.pos, 1);
|
||||||
|
expand(cm, known.region);
|
||||||
|
} else {
|
||||||
|
var end = rangeFinder(cm, line, hideEnd);
|
||||||
|
if (end == null) return;
|
||||||
|
var hidden = [];
|
||||||
|
for (var i = line + 1; i < end; ++i) {
|
||||||
|
var handle = cm.hideLine(i);
|
||||||
|
if (handle) hidden.push(handle);
|
||||||
|
}
|
||||||
|
var first = cm.setMarker(line, markText);
|
||||||
|
var region = {start: first, hidden: hidden};
|
||||||
|
cm.onDeleteLine(first, function() { expand(cm, region); });
|
||||||
|
folded.push(region);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
};
|
196
codemirror-2.36/lib/util/formatting.js
Normal file
196
codemirror-2.36/lib/util/formatting.js
Normal file
|
@ -0,0 +1,196 @@
|
||||||
|
// ============== Formatting extensions ============================
|
||||||
|
(function() {
|
||||||
|
// Define extensions for a few modes
|
||||||
|
CodeMirror.extendMode("css", {
|
||||||
|
commentStart: "/*",
|
||||||
|
commentEnd: "*/",
|
||||||
|
wordWrapChars: [";", "\\{", "\\}"],
|
||||||
|
autoFormatLineBreaks: function (text) {
|
||||||
|
return text.replace(new RegExp("(;|\\{|\\})([^\r\n])", "g"), "$1\n$2");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function jsNonBreakableBlocks(text) {
|
||||||
|
var nonBreakableRegexes = [/for\s*?\((.*?)\)/,
|
||||||
|
/\"(.*?)(\"|$)/,
|
||||||
|
/\'(.*?)(\'|$)/,
|
||||||
|
/\/\*(.*?)(\*\/|$)/,
|
||||||
|
/\/\/.*/];
|
||||||
|
var nonBreakableBlocks = [];
|
||||||
|
for (var i = 0; i < nonBreakableRegexes.length; i++) {
|
||||||
|
var curPos = 0;
|
||||||
|
while (curPos < text.length) {
|
||||||
|
var m = text.substr(curPos).match(nonBreakableRegexes[i]);
|
||||||
|
if (m != null) {
|
||||||
|
nonBreakableBlocks.push({
|
||||||
|
start: curPos + m.index,
|
||||||
|
end: curPos + m.index + m[0].length
|
||||||
|
});
|
||||||
|
curPos += m.index + Math.max(1, m[0].length);
|
||||||
|
}
|
||||||
|
else { // No more matches
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nonBreakableBlocks.sort(function (a, b) {
|
||||||
|
return a.start - b.start;
|
||||||
|
});
|
||||||
|
|
||||||
|
return nonBreakableBlocks;
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.extendMode("javascript", {
|
||||||
|
commentStart: "/*",
|
||||||
|
commentEnd: "*/",
|
||||||
|
wordWrapChars: [";", "\\{", "\\}"],
|
||||||
|
|
||||||
|
autoFormatLineBreaks: function (text) {
|
||||||
|
var curPos = 0;
|
||||||
|
var split = this.jsonMode ? function(str) {
|
||||||
|
return str.replace(/([,{])/g, "$1\n").replace(/}/g, "\n}");
|
||||||
|
} : function(str) {
|
||||||
|
return str.replace(/(;|\{|\})([^\r\n;])/g, "$1\n$2");
|
||||||
|
};
|
||||||
|
var nonBreakableBlocks = jsNonBreakableBlocks(text), res = "";
|
||||||
|
if (nonBreakableBlocks != null) {
|
||||||
|
for (var i = 0; i < nonBreakableBlocks.length; i++) {
|
||||||
|
if (nonBreakableBlocks[i].start > curPos) { // Break lines till the block
|
||||||
|
res += split(text.substring(curPos, nonBreakableBlocks[i].start));
|
||||||
|
curPos = nonBreakableBlocks[i].start;
|
||||||
|
}
|
||||||
|
if (nonBreakableBlocks[i].start <= curPos
|
||||||
|
&& nonBreakableBlocks[i].end >= curPos) { // Skip non-breakable block
|
||||||
|
res += text.substring(curPos, nonBreakableBlocks[i].end);
|
||||||
|
curPos = nonBreakableBlocks[i].end;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (curPos < text.length)
|
||||||
|
res += split(text.substr(curPos));
|
||||||
|
} else {
|
||||||
|
res = split(text);
|
||||||
|
}
|
||||||
|
return res.replace(/^\n*|\n*$/, "");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.extendMode("xml", {
|
||||||
|
commentStart: "<!--",
|
||||||
|
commentEnd: "-->",
|
||||||
|
wordWrapChars: [">"],
|
||||||
|
|
||||||
|
autoFormatLineBreaks: function (text) {
|
||||||
|
var lines = text.split("\n");
|
||||||
|
var reProcessedPortion = new RegExp("(^\\s*?<|^[^<]*?)(.+)(>\\s*?$|[^>]*?$)");
|
||||||
|
var reOpenBrackets = new RegExp("<", "g");
|
||||||
|
var reCloseBrackets = new RegExp("(>)([^\r\n])", "g");
|
||||||
|
for (var i = 0; i < lines.length; i++) {
|
||||||
|
var mToProcess = lines[i].match(reProcessedPortion);
|
||||||
|
if (mToProcess != null && mToProcess.length > 3) { // The line starts with whitespaces and ends with whitespaces
|
||||||
|
lines[i] = mToProcess[1]
|
||||||
|
+ mToProcess[2].replace(reOpenBrackets, "\n$&").replace(reCloseBrackets, "$1\n$2")
|
||||||
|
+ mToProcess[3];
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lines.join("\n");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function localModeAt(cm, pos) {
|
||||||
|
return CodeMirror.innerMode(cm.getMode(), cm.getTokenAt(pos).state).mode;
|
||||||
|
}
|
||||||
|
|
||||||
|
function enumerateModesBetween(cm, line, start, end) {
|
||||||
|
var outer = cm.getMode(), text = cm.getLine(line);
|
||||||
|
if (end == null) end = text.length;
|
||||||
|
if (!outer.innerMode) return [{from: start, to: end, mode: outer}];
|
||||||
|
var state = cm.getTokenAt({line: line, ch: start}).state;
|
||||||
|
var mode = CodeMirror.innerMode(outer, state).mode;
|
||||||
|
var found = [], stream = new CodeMirror.StringStream(text);
|
||||||
|
stream.pos = stream.start = start;
|
||||||
|
for (;;) {
|
||||||
|
outer.token(stream, state);
|
||||||
|
var curMode = CodeMirror.innerMode(outer, state).mode;
|
||||||
|
if (curMode != mode) {
|
||||||
|
var cut = stream.start;
|
||||||
|
// Crappy heuristic to deal with the fact that a change in
|
||||||
|
// mode can occur both at the end and the start of a token,
|
||||||
|
// and we don't know which it was.
|
||||||
|
if (mode.name == "xml" && text.charAt(stream.pos - 1) == ">") cut = stream.pos;
|
||||||
|
found.push({from: start, to: cut, mode: mode});
|
||||||
|
start = cut;
|
||||||
|
mode = curMode;
|
||||||
|
}
|
||||||
|
if (stream.pos >= end) break;
|
||||||
|
stream.start = stream.pos;
|
||||||
|
}
|
||||||
|
if (start < end) found.push({from: start, to: end, mode: mode});
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Comment/uncomment the specified range
|
||||||
|
CodeMirror.defineExtension("commentRange", function (isComment, from, to) {
|
||||||
|
var curMode = localModeAt(this, from), cm = this;
|
||||||
|
this.operation(function() {
|
||||||
|
if (isComment) { // Comment range
|
||||||
|
cm.replaceRange(curMode.commentEnd, to);
|
||||||
|
cm.replaceRange(curMode.commentStart, from);
|
||||||
|
if (from.line == to.line && from.ch == to.ch) // An empty comment inserted - put cursor inside
|
||||||
|
cm.setCursor(from.line, from.ch + curMode.commentStart.length);
|
||||||
|
} else { // Uncomment range
|
||||||
|
var selText = cm.getRange(from, to);
|
||||||
|
var startIndex = selText.indexOf(curMode.commentStart);
|
||||||
|
var endIndex = selText.lastIndexOf(curMode.commentEnd);
|
||||||
|
if (startIndex > -1 && endIndex > -1 && endIndex > startIndex) {
|
||||||
|
// Take string till comment start
|
||||||
|
selText = selText.substr(0, startIndex)
|
||||||
|
// From comment start till comment end
|
||||||
|
+ selText.substring(startIndex + curMode.commentStart.length, endIndex)
|
||||||
|
// From comment end till string end
|
||||||
|
+ selText.substr(endIndex + curMode.commentEnd.length);
|
||||||
|
}
|
||||||
|
cm.replaceRange(selText, from, to);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Applies automatic mode-aware indentation to the specified range
|
||||||
|
CodeMirror.defineExtension("autoIndentRange", function (from, to) {
|
||||||
|
var cmInstance = this;
|
||||||
|
this.operation(function () {
|
||||||
|
for (var i = from.line; i <= to.line; i++) {
|
||||||
|
cmInstance.indentLine(i, "smart");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Applies automatic formatting to the specified range
|
||||||
|
CodeMirror.defineExtension("autoFormatRange", function (from, to) {
|
||||||
|
var cm = this;
|
||||||
|
cm.operation(function () {
|
||||||
|
for (var cur = from.line, end = to.line; cur <= end; ++cur) {
|
||||||
|
var f = {line: cur, ch: cur == from.line ? from.ch : 0};
|
||||||
|
var t = {line: cur, ch: cur == end ? to.ch : null};
|
||||||
|
var modes = enumerateModesBetween(cm, cur, f.ch, t.ch), mangled = "";
|
||||||
|
var text = cm.getRange(f, t);
|
||||||
|
for (var i = 0; i < modes.length; ++i) {
|
||||||
|
var part = modes.length > 1 ? text.slice(modes[i].from, modes[i].to) : text;
|
||||||
|
if (mangled) mangled += "\n";
|
||||||
|
if (modes[i].mode.autoFormatLineBreaks) {
|
||||||
|
mangled += modes[i].mode.autoFormatLineBreaks(part);
|
||||||
|
} else mangled += text;
|
||||||
|
}
|
||||||
|
if (mangled != text) {
|
||||||
|
for (var count = 0, pos = mangled.indexOf("\n"); pos != -1; pos = mangled.indexOf("\n", pos + 1), ++count) {}
|
||||||
|
cm.replaceRange(mangled, f, t);
|
||||||
|
cur += count;
|
||||||
|
end += count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (var cur = from.line + 1; cur <= end; ++cur)
|
||||||
|
cm.indentLine(cur, "smart");
|
||||||
|
cm.setSelection(from, cm.getCursor(false));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})();
|
134
codemirror-2.36/lib/util/javascript-hint.js
Normal file
134
codemirror-2.36/lib/util/javascript-hint.js
Normal file
|
@ -0,0 +1,134 @@
|
||||||
|
(function () {
|
||||||
|
function forEach(arr, f) {
|
||||||
|
for (var i = 0, e = arr.length; i < e; ++i) f(arr[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function arrayContains(arr, item) {
|
||||||
|
if (!Array.prototype.indexOf) {
|
||||||
|
var i = arr.length;
|
||||||
|
while (i--) {
|
||||||
|
if (arr[i] === item) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return arr.indexOf(item) != -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
function scriptHint(editor, keywords, getToken) {
|
||||||
|
// Find the token at the cursor
|
||||||
|
var cur = editor.getCursor(), token = getToken(editor, cur), tprop = token;
|
||||||
|
// If it's not a 'word-style' token, ignore the token.
|
||||||
|
if (!/^[\w$_]*$/.test(token.string)) {
|
||||||
|
token = tprop = {start: cur.ch, end: cur.ch, string: "", state: token.state,
|
||||||
|
className: token.string == "." ? "property" : null};
|
||||||
|
}
|
||||||
|
// If it is a property, find out what it is a property of.
|
||||||
|
while (tprop.className == "property") {
|
||||||
|
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
||||||
|
if (tprop.string != ".") return;
|
||||||
|
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
||||||
|
if (tprop.string == ')') {
|
||||||
|
var level = 1;
|
||||||
|
do {
|
||||||
|
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
||||||
|
switch (tprop.string) {
|
||||||
|
case ')': level++; break;
|
||||||
|
case '(': level--; break;
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
|
} while (level > 0);
|
||||||
|
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
||||||
|
if (tprop.className == 'variable')
|
||||||
|
tprop.className = 'function';
|
||||||
|
else return; // no clue
|
||||||
|
}
|
||||||
|
if (!context) var context = [];
|
||||||
|
context.push(tprop);
|
||||||
|
}
|
||||||
|
return {list: getCompletions(token, context, keywords),
|
||||||
|
from: {line: cur.line, ch: token.start},
|
||||||
|
to: {line: cur.line, ch: token.end}};
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.javascriptHint = function(editor) {
|
||||||
|
return scriptHint(editor, javascriptKeywords,
|
||||||
|
function (e, cur) {return e.getTokenAt(cur);});
|
||||||
|
};
|
||||||
|
|
||||||
|
function getCoffeeScriptToken(editor, cur) {
|
||||||
|
// This getToken, it is for coffeescript, imitates the behavior of
|
||||||
|
// getTokenAt method in javascript.js, that is, returning "property"
|
||||||
|
// type and treat "." as indepenent token.
|
||||||
|
var token = editor.getTokenAt(cur);
|
||||||
|
if (cur.ch == token.start + 1 && token.string.charAt(0) == '.') {
|
||||||
|
token.end = token.start;
|
||||||
|
token.string = '.';
|
||||||
|
token.className = "property";
|
||||||
|
}
|
||||||
|
else if (/^\.[\w$_]*$/.test(token.string)) {
|
||||||
|
token.className = "property";
|
||||||
|
token.start++;
|
||||||
|
token.string = token.string.replace(/\./, '');
|
||||||
|
}
|
||||||
|
return token;
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.coffeescriptHint = function(editor) {
|
||||||
|
return scriptHint(editor, coffeescriptKeywords, getCoffeeScriptToken);
|
||||||
|
};
|
||||||
|
|
||||||
|
var stringProps = ("charAt charCodeAt indexOf lastIndexOf substring substr slice trim trimLeft trimRight " +
|
||||||
|
"toUpperCase toLowerCase split concat match replace search").split(" ");
|
||||||
|
var arrayProps = ("length concat join splice push pop shift unshift slice reverse sort indexOf " +
|
||||||
|
"lastIndexOf every some filter forEach map reduce reduceRight ").split(" ");
|
||||||
|
var funcProps = "prototype apply call bind".split(" ");
|
||||||
|
var javascriptKeywords = ("break case catch continue debugger default delete do else false finally for function " +
|
||||||
|
"if in instanceof new null return switch throw true try typeof var void while with").split(" ");
|
||||||
|
var coffeescriptKeywords = ("and break catch class continue delete do else extends false finally for " +
|
||||||
|
"if in instanceof isnt new no not null of off on or return switch then throw true try typeof until void while with yes").split(" ");
|
||||||
|
|
||||||
|
function getCompletions(token, context, keywords) {
|
||||||
|
var found = [], start = token.string;
|
||||||
|
function maybeAdd(str) {
|
||||||
|
if (str.indexOf(start) == 0 && !arrayContains(found, str)) found.push(str);
|
||||||
|
}
|
||||||
|
function gatherCompletions(obj) {
|
||||||
|
if (typeof obj == "string") forEach(stringProps, maybeAdd);
|
||||||
|
else if (obj instanceof Array) forEach(arrayProps, maybeAdd);
|
||||||
|
else if (obj instanceof Function) forEach(funcProps, maybeAdd);
|
||||||
|
for (var name in obj) maybeAdd(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (context) {
|
||||||
|
// If this is a property, see if it belongs to some object we can
|
||||||
|
// find in the current environment.
|
||||||
|
var obj = context.pop(), base;
|
||||||
|
if (obj.className == "variable")
|
||||||
|
base = window[obj.string];
|
||||||
|
else if (obj.className == "string")
|
||||||
|
base = "";
|
||||||
|
else if (obj.className == "atom")
|
||||||
|
base = 1;
|
||||||
|
else if (obj.className == "function") {
|
||||||
|
if (window.jQuery != null && (obj.string == '$' || obj.string == 'jQuery') &&
|
||||||
|
(typeof window.jQuery == 'function'))
|
||||||
|
base = window.jQuery();
|
||||||
|
else if (window._ != null && (obj.string == '_') && (typeof window._ == 'function'))
|
||||||
|
base = window._();
|
||||||
|
}
|
||||||
|
while (base != null && context.length)
|
||||||
|
base = base[context.pop().string];
|
||||||
|
if (base != null) gatherCompletions(base);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// If not, just look in the window object and any local scope
|
||||||
|
// (reading into JS mode internals to get at the local variables)
|
||||||
|
for (var v = token.state.localVars; v; v = v.next) maybeAdd(v.name);
|
||||||
|
gatherCompletions(window);
|
||||||
|
forEach(keywords, maybeAdd);
|
||||||
|
}
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
})();
|
51
codemirror-2.36/lib/util/loadmode.js
Normal file
51
codemirror-2.36/lib/util/loadmode.js
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
(function() {
|
||||||
|
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 script = document.createElement("script");
|
||||||
|
script.src = CodeMirror.modeURL.replace(/%N/g, mode);
|
||||||
|
var others = document.getElementsByTagName("script")[0];
|
||||||
|
others.parentNode.insertBefore(script, others);
|
||||||
|
var list = loading[mode] = [cont];
|
||||||
|
var count = 0, poll = setInterval(function() {
|
||||||
|
if (++count > 100) return clearInterval(poll);
|
||||||
|
if (CodeMirror.modes.hasOwnProperty(mode)) {
|
||||||
|
clearInterval(poll);
|
||||||
|
loading[mode] = null;
|
||||||
|
ensureDeps(mode, function() {
|
||||||
|
for (var i = 0; i < list.length; ++i) list[i]();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 200);
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.autoLoadMode = function(instance, mode) {
|
||||||
|
if (!CodeMirror.modes.hasOwnProperty(mode))
|
||||||
|
CodeMirror.requireMode(mode, function() {
|
||||||
|
instance.setOption("mode", instance.getOption("mode"));
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}());
|
44
codemirror-2.36/lib/util/match-highlighter.js
Normal file
44
codemirror-2.36/lib/util/match-highlighter.js
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
// Define match-highlighter commands. Depends on searchcursor.js
|
||||||
|
// Use by attaching the following function call to the onCursorActivity event:
|
||||||
|
//myCodeMirror.matchHighlight(minChars);
|
||||||
|
// And including a special span.CodeMirror-matchhighlight css class (also optionally a separate one for .CodeMirror-focused -- see demo matchhighlighter.html)
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
var DEFAULT_MIN_CHARS = 2;
|
||||||
|
|
||||||
|
function MatchHighlightState() {
|
||||||
|
this.marked = [];
|
||||||
|
}
|
||||||
|
function getMatchHighlightState(cm) {
|
||||||
|
return cm._matchHighlightState || (cm._matchHighlightState = new MatchHighlightState());
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearMarks(cm) {
|
||||||
|
var state = getMatchHighlightState(cm);
|
||||||
|
for (var i = 0; i < state.marked.length; ++i)
|
||||||
|
state.marked[i].clear();
|
||||||
|
state.marked = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
function markDocument(cm, className, minChars) {
|
||||||
|
clearMarks(cm);
|
||||||
|
minChars = (typeof minChars !== 'undefined' ? minChars : DEFAULT_MIN_CHARS);
|
||||||
|
if (cm.somethingSelected() && cm.getSelection().replace(/^\s+|\s+$/g, "").length >= minChars) {
|
||||||
|
var state = getMatchHighlightState(cm);
|
||||||
|
var query = cm.getSelection();
|
||||||
|
cm.operation(function() {
|
||||||
|
if (cm.lineCount() < 2000) { // This is too expensive on big documents.
|
||||||
|
for (var cursor = cm.getSearchCursor(query); cursor.findNext();) {
|
||||||
|
//Only apply matchhighlight to the matches other than the one actually selected
|
||||||
|
if (!(cursor.from().line === cm.getCursor(true).line && cursor.from().ch === cm.getCursor(true).ch))
|
||||||
|
state.marked.push(cm.markText(cursor.from(), cursor.to(), className));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.defineExtension("matchHighlight", function(className, minChars) {
|
||||||
|
markDocument(this, className, minChars);
|
||||||
|
});
|
||||||
|
})();
|
77
codemirror-2.36/lib/util/multiplex.js
Normal file
77
codemirror-2.36/lib/util/multiplex.js
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
CodeMirror.multiplexingMode = function(outer /*, others */) {
|
||||||
|
// Others should be {open, close, mode [, delimStyle]} objects
|
||||||
|
var others = Array.prototype.slice.call(arguments, 1);
|
||||||
|
var n_others = others.length;
|
||||||
|
|
||||||
|
function indexOf(string, pattern, from) {
|
||||||
|
if (typeof pattern == "string") return string.indexOf(pattern, from);
|
||||||
|
var m = pattern.exec(from ? string.slice(from) : string);
|
||||||
|
return m ? m.index + from : -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 < n_others; ++i) {
|
||||||
|
var other = others[i];
|
||||||
|
var found = indexOf(oldContent, other.open, stream.pos);
|
||||||
|
if (found == stream.pos) {
|
||||||
|
stream.match(other.open);
|
||||||
|
state.innerActive = other;
|
||||||
|
state.inner = CodeMirror.startState(other.mode, outer.indent ? outer.indent(state.outer, "") : 0);
|
||||||
|
return other.delimStyle;
|
||||||
|
} 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;
|
||||||
|
var found = indexOf(oldContent, curInner.close, stream.pos);
|
||||||
|
if (found == stream.pos) {
|
||||||
|
stream.match(curInner.close);
|
||||||
|
state.innerActive = state.inner = null;
|
||||||
|
return curInner.delimStyle;
|
||||||
|
}
|
||||||
|
if (found > -1) stream.string = oldContent.slice(0, found);
|
||||||
|
var innerToken = curInner.mode.token(stream, state.inner);
|
||||||
|
if (found > -1) stream.string = oldContent;
|
||||||
|
var cur = stream.current(), found = cur.indexOf(curInner.close);
|
||||||
|
if (found > -1) stream.backUp(cur.length - found);
|
||||||
|
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);
|
||||||
|
},
|
||||||
|
|
||||||
|
electricChars: outer.electricChars,
|
||||||
|
|
||||||
|
innerMode: function(state) {
|
||||||
|
return state.inner ? {state: state.inner, mode: state.innerActive.mode} : {state: state.outer, mode: outer};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
59
codemirror-2.36/lib/util/overlay.js
Normal file
59
codemirror-2.36/lib/util/overlay.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
// 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, in which case
|
||||||
|
// the styles are combined.
|
||||||
|
|
||||||
|
// overlayParser is the old, deprecated name
|
||||||
|
CodeMirror.overlayMode = CodeMirror.overlayParser = function(base, overlay, combine) {
|
||||||
|
return {
|
||||||
|
startState: function() {
|
||||||
|
return {
|
||||||
|
base: CodeMirror.startState(base),
|
||||||
|
overlay: CodeMirror.startState(overlay),
|
||||||
|
basePos: 0, baseCur: null,
|
||||||
|
overlayPos: 0, overlayCur: 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.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);
|
||||||
|
if (stream.eol()) state.basePos = state.overlayPos = 0;
|
||||||
|
|
||||||
|
if (state.overlayCur == null) return state.baseCur;
|
||||||
|
if (state.baseCur != null && combine) 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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
123
codemirror-2.36/lib/util/pig-hint.js
Normal file
123
codemirror-2.36/lib/util/pig-hint.js
Normal file
|
@ -0,0 +1,123 @@
|
||||||
|
(function () {
|
||||||
|
function forEach(arr, f) {
|
||||||
|
for (var i = 0, e = arr.length; i < e; ++i) f(arr[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function arrayContains(arr, item) {
|
||||||
|
if (!Array.prototype.indexOf) {
|
||||||
|
var i = arr.length;
|
||||||
|
while (i--) {
|
||||||
|
if (arr[i] === item) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return arr.indexOf(item) != -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
function scriptHint(editor, keywords, getToken) {
|
||||||
|
// Find the token at the cursor
|
||||||
|
var cur = editor.getCursor(), token = getToken(editor, cur), tprop = token;
|
||||||
|
// If it's not a 'word-style' token, ignore the token.
|
||||||
|
|
||||||
|
if (!/^[\w$_]*$/.test(token.string)) {
|
||||||
|
token = tprop = {start: cur.ch, end: cur.ch, string: "", state: token.state,
|
||||||
|
className: token.string == ":" ? "pig-type" : null};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!context) var context = [];
|
||||||
|
context.push(tprop);
|
||||||
|
|
||||||
|
var completionList = getCompletions(token, context);
|
||||||
|
completionList = completionList.sort();
|
||||||
|
//prevent autocomplete for last word, instead show dropdown with one word
|
||||||
|
if(completionList.length == 1) {
|
||||||
|
completionList.push(" ");
|
||||||
|
}
|
||||||
|
|
||||||
|
return {list: completionList,
|
||||||
|
from: {line: cur.line, ch: token.start},
|
||||||
|
to: {line: cur.line, ch: token.end}};
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.pigHint = function(editor) {
|
||||||
|
return scriptHint(editor, pigKeywordsU, function (e, cur) {return e.getTokenAt(cur);});
|
||||||
|
};
|
||||||
|
|
||||||
|
function toTitleCase(str) {
|
||||||
|
return str.replace(/(?:^|\s)\w/g, function(match) {
|
||||||
|
return match.toUpperCase();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var pigKeywords = "VOID IMPORT RETURNS DEFINE LOAD FILTER FOREACH ORDER CUBE DISTINCT COGROUP "
|
||||||
|
+ "JOIN CROSS UNION SPLIT INTO IF OTHERWISE ALL AS BY USING INNER OUTER ONSCHEMA PARALLEL "
|
||||||
|
+ "PARTITION GROUP AND OR NOT GENERATE FLATTEN ASC DESC IS STREAM THROUGH STORE MAPREDUCE "
|
||||||
|
+ "SHIP CACHE INPUT OUTPUT STDERROR STDIN STDOUT LIMIT SAMPLE LEFT RIGHT FULL EQ GT LT GTE LTE "
|
||||||
|
+ "NEQ MATCHES TRUE FALSE";
|
||||||
|
var pigKeywordsU = pigKeywords.split(" ");
|
||||||
|
var pigKeywordsL = pigKeywords.toLowerCase().split(" ");
|
||||||
|
|
||||||
|
var pigTypes = "BOOLEAN INT LONG FLOAT DOUBLE CHARARRAY BYTEARRAY BAG TUPLE MAP";
|
||||||
|
var pigTypesU = pigTypes.split(" ");
|
||||||
|
var pigTypesL = pigTypes.toLowerCase().split(" ");
|
||||||
|
|
||||||
|
var pigBuiltins = "ABS ACOS ARITY ASIN ATAN AVG BAGSIZE BINSTORAGE BLOOM BUILDBLOOM CBRT CEIL "
|
||||||
|
+ "CONCAT COR COS COSH COUNT COUNT_STAR COV CONSTANTSIZE CUBEDIMENSIONS DIFF DISTINCT DOUBLEABS "
|
||||||
|
+ "DOUBLEAVG DOUBLEBASE DOUBLEMAX DOUBLEMIN DOUBLEROUND DOUBLESUM EXP FLOOR FLOATABS FLOATAVG "
|
||||||
|
+ "FLOATMAX FLOATMIN FLOATROUND FLOATSUM GENERICINVOKER INDEXOF INTABS INTAVG INTMAX INTMIN "
|
||||||
|
+ "INTSUM INVOKEFORDOUBLE INVOKEFORFLOAT INVOKEFORINT INVOKEFORLONG INVOKEFORSTRING INVOKER "
|
||||||
|
+ "ISEMPTY JSONLOADER JSONMETADATA JSONSTORAGE LAST_INDEX_OF LCFIRST LOG LOG10 LOWER LONGABS "
|
||||||
|
+ "LONGAVG LONGMAX LONGMIN LONGSUM MAX MIN MAPSIZE MONITOREDUDF NONDETERMINISTIC OUTPUTSCHEMA "
|
||||||
|
+ "PIGSTORAGE PIGSTREAMING RANDOM REGEX_EXTRACT REGEX_EXTRACT_ALL REPLACE ROUND SIN SINH SIZE "
|
||||||
|
+ "SQRT STRSPLIT SUBSTRING SUM STRINGCONCAT STRINGMAX STRINGMIN STRINGSIZE TAN TANH TOBAG "
|
||||||
|
+ "TOKENIZE TOMAP TOP TOTUPLE TRIM TEXTLOADER TUPLESIZE UCFIRST UPPER UTF8STORAGECONVERTER";
|
||||||
|
var pigBuiltinsU = pigBuiltins.split(" ").join("() ").split(" ");
|
||||||
|
var pigBuiltinsL = pigBuiltins.toLowerCase().split(" ").join("() ").split(" ");
|
||||||
|
var pigBuiltinsC = ("BagSize BinStorage Bloom BuildBloom ConstantSize CubeDimensions DoubleAbs "
|
||||||
|
+ "DoubleAvg DoubleBase DoubleMax DoubleMin DoubleRound DoubleSum FloatAbs FloatAvg FloatMax "
|
||||||
|
+ "FloatMin FloatRound FloatSum GenericInvoker IntAbs IntAvg IntMax IntMin IntSum "
|
||||||
|
+ "InvokeForDouble InvokeForFloat InvokeForInt InvokeForLong InvokeForString Invoker "
|
||||||
|
+ "IsEmpty JsonLoader JsonMetadata JsonStorage LongAbs LongAvg LongMax LongMin LongSum MapSize "
|
||||||
|
+ "MonitoredUDF Nondeterministic OutputSchema PigStorage PigStreaming StringConcat StringMax "
|
||||||
|
+ "StringMin StringSize TextLoader TupleSize Utf8StorageConverter").split(" ").join("() ").split(" ");
|
||||||
|
|
||||||
|
function getCompletions(token, context) {
|
||||||
|
var found = [], start = token.string;
|
||||||
|
function maybeAdd(str) {
|
||||||
|
if (str.indexOf(start) == 0 && !arrayContains(found, str)) found.push(str);
|
||||||
|
}
|
||||||
|
|
||||||
|
function gatherCompletions(obj) {
|
||||||
|
if(obj == ":") {
|
||||||
|
forEach(pigTypesL, maybeAdd);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
forEach(pigBuiltinsU, maybeAdd);
|
||||||
|
forEach(pigBuiltinsL, maybeAdd);
|
||||||
|
forEach(pigBuiltinsC, maybeAdd);
|
||||||
|
forEach(pigTypesU, maybeAdd);
|
||||||
|
forEach(pigTypesL, maybeAdd);
|
||||||
|
forEach(pigKeywordsU, maybeAdd);
|
||||||
|
forEach(pigKeywordsL, maybeAdd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (context) {
|
||||||
|
// If this is a property, see if it belongs to some object we can
|
||||||
|
// find in the current environment.
|
||||||
|
var obj = context.pop(), base;
|
||||||
|
|
||||||
|
if (obj.className == "pig-word")
|
||||||
|
base = obj.string;
|
||||||
|
else if(obj.className == "pig-type")
|
||||||
|
base = ":" + obj.string;
|
||||||
|
|
||||||
|
while (base != null && context.length)
|
||||||
|
base = base[context.pop().string];
|
||||||
|
if (base != null) gatherCompletions(base);
|
||||||
|
}
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
})();
|
90
codemirror-2.36/lib/util/runmode-standalone.js
Normal file
90
codemirror-2.36/lib/util/runmode-standalone.js
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
/* Just enough of CodeMirror to run runMode under node.js */
|
||||||
|
|
||||||
|
function splitLines(string){ return string.split(/\r?\n|\r/); };
|
||||||
|
|
||||||
|
function StringStream(string) {
|
||||||
|
this.pos = this.start = 0;
|
||||||
|
this.string = string;
|
||||||
|
}
|
||||||
|
StringStream.prototype = {
|
||||||
|
eol: function() {return this.pos >= this.string.length;},
|
||||||
|
sol: function() {return this.pos == 0;},
|
||||||
|
peek: function() {return this.string.charAt(this.pos) || null;},
|
||||||
|
next: function() {
|
||||||
|
if (this.pos < this.string.length)
|
||||||
|
return this.string.charAt(this.pos++);
|
||||||
|
},
|
||||||
|
eat: function(match) {
|
||||||
|
var ch = this.string.charAt(this.pos);
|
||||||
|
if (typeof match == "string") var ok = ch == match;
|
||||||
|
else var ok = ch && (match.test ? match.test(ch) : match(ch));
|
||||||
|
if (ok) {++this.pos; return ch;}
|
||||||
|
},
|
||||||
|
eatWhile: function(match) {
|
||||||
|
var start = this.pos;
|
||||||
|
while (this.eat(match)){}
|
||||||
|
return this.pos > start;
|
||||||
|
},
|
||||||
|
eatSpace: function() {
|
||||||
|
var start = this.pos;
|
||||||
|
while (/[\s\u00a0]/.test(this.string.charAt(this.pos))) ++this.pos;
|
||||||
|
return this.pos > start;
|
||||||
|
},
|
||||||
|
skipToEnd: function() {this.pos = this.string.length;},
|
||||||
|
skipTo: function(ch) {
|
||||||
|
var found = this.string.indexOf(ch, this.pos);
|
||||||
|
if (found > -1) {this.pos = found; return true;}
|
||||||
|
},
|
||||||
|
backUp: function(n) {this.pos -= n;},
|
||||||
|
column: function() {return this.start;},
|
||||||
|
indentation: function() {return 0;},
|
||||||
|
match: function(pattern, consume, caseInsensitive) {
|
||||||
|
if (typeof pattern == "string") {
|
||||||
|
function cased(str) {return caseInsensitive ? str.toLowerCase() : str;}
|
||||||
|
if (cased(this.string).indexOf(cased(pattern), this.pos) == this.pos) {
|
||||||
|
if (consume !== false) this.pos += pattern.length;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var match = this.string.slice(this.pos).match(pattern);
|
||||||
|
if (match && consume !== false) this.pos += match[0].length;
|
||||||
|
return match;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
current: function(){return this.string.slice(this.start, this.pos);}
|
||||||
|
};
|
||||||
|
exports.StringStream = StringStream;
|
||||||
|
|
||||||
|
exports.startState = function(mode, a1, a2) {
|
||||||
|
return mode.startState ? mode.startState(a1, a2) : true;
|
||||||
|
};
|
||||||
|
|
||||||
|
var modes = exports.modes = {}, mimeModes = exports.mimeModes = {};
|
||||||
|
exports.defineMode = function(name, mode) { modes[name] = mode; };
|
||||||
|
exports.defineMIME = function(mime, spec) { mimeModes[mime] = spec; };
|
||||||
|
exports.getMode = function(options, spec) {
|
||||||
|
if (typeof spec == "string" && mimeModes.hasOwnProperty(spec))
|
||||||
|
spec = mimeModes[spec];
|
||||||
|
if (typeof spec == "string")
|
||||||
|
var mname = spec, config = {};
|
||||||
|
else if (spec != null)
|
||||||
|
var mname = spec.name, config = spec;
|
||||||
|
var mfactory = modes[mname];
|
||||||
|
if (!mfactory) throw new Error("Unknown mode: " + spec);
|
||||||
|
return mfactory(options, config || {});
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.runMode = function(string, modespec, callback) {
|
||||||
|
var mode = exports.getMode({indentUnit: 2}, modespec);
|
||||||
|
var lines = splitLines(string), state = exports.startState(mode);
|
||||||
|
for (var i = 0, e = lines.length; i < e; ++i) {
|
||||||
|
if (i) callback("\n");
|
||||||
|
var stream = new exports.StringStream(lines[i]);
|
||||||
|
while (!stream.eol()) {
|
||||||
|
var style = mode.token(stream, state);
|
||||||
|
callback(stream.current(), style, i, stream.start);
|
||||||
|
stream.start = stream.pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
53
codemirror-2.36/lib/util/runmode.js
Normal file
53
codemirror-2.36/lib/util/runmode.js
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
CodeMirror.runMode = function(string, modespec, callback, options) {
|
||||||
|
function esc(str) {
|
||||||
|
return str.replace(/[<&]/g, function(ch) { return ch == "<" ? "<" : "&"; });
|
||||||
|
}
|
||||||
|
|
||||||
|
var mode = CodeMirror.getMode(CodeMirror.defaults, modespec);
|
||||||
|
var isNode = callback.nodeType == 1;
|
||||||
|
var tabSize = (options && options.tabSize) || CodeMirror.defaults.tabSize;
|
||||||
|
if (isNode) {
|
||||||
|
var node = callback, accum = [], col = 0;
|
||||||
|
callback = function(text, style) {
|
||||||
|
if (text == "\n") {
|
||||||
|
accum.push("<br>");
|
||||||
|
col = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var escaped = "";
|
||||||
|
// HTML-escape and replace tabs
|
||||||
|
for (var pos = 0;;) {
|
||||||
|
var idx = text.indexOf("\t", pos);
|
||||||
|
if (idx == -1) {
|
||||||
|
escaped += esc(text.slice(pos));
|
||||||
|
col += text.length - pos;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
col += idx - pos;
|
||||||
|
escaped += esc(text.slice(pos, idx));
|
||||||
|
var size = tabSize - col % tabSize;
|
||||||
|
col += size;
|
||||||
|
for (var i = 0; i < size; ++i) escaped += " ";
|
||||||
|
pos = idx + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style)
|
||||||
|
accum.push("<span class=\"cm-" + esc(style) + "\">" + escaped + "</span>");
|
||||||
|
else
|
||||||
|
accum.push(escaped);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
var lines = CodeMirror.splitLines(string), state = CodeMirror.startState(mode);
|
||||||
|
for (var i = 0, e = lines.length; i < e; ++i) {
|
||||||
|
if (i) callback("\n");
|
||||||
|
var stream = new CodeMirror.StringStream(lines[i]);
|
||||||
|
while (!stream.eol()) {
|
||||||
|
var style = mode.token(stream, state);
|
||||||
|
callback(stream.current(), style, i, stream.start);
|
||||||
|
stream.start = stream.pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (isNode)
|
||||||
|
node.innerHTML = accum.join("");
|
||||||
|
};
|
118
codemirror-2.36/lib/util/search.js
Normal file
118
codemirror-2.36/lib/util/search.js
Normal file
|
@ -0,0 +1,118 @@
|
||||||
|
// Define search commands. Depends on dialog.js or another
|
||||||
|
// implementation of the openDialog method.
|
||||||
|
|
||||||
|
// Replace works a little oddly -- it will do the replace on the next
|
||||||
|
// Ctrl-G (or whatever is bound to findNext) press. You prevent a
|
||||||
|
// replace by making sure the match is no longer selected when hitting
|
||||||
|
// Ctrl-G.
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
function SearchState() {
|
||||||
|
this.posFrom = this.posTo = this.query = null;
|
||||||
|
this.marked = [];
|
||||||
|
}
|
||||||
|
function getSearchState(cm) {
|
||||||
|
return cm._searchState || (cm._searchState = new SearchState());
|
||||||
|
}
|
||||||
|
function getSearchCursor(cm, query, pos) {
|
||||||
|
// Heuristic: if the query string is all lowercase, do a case insensitive search.
|
||||||
|
return cm.getSearchCursor(query, pos, typeof query == "string" && query == query.toLowerCase());
|
||||||
|
}
|
||||||
|
function dialog(cm, text, shortText, f) {
|
||||||
|
if (cm.openDialog) cm.openDialog(text, f);
|
||||||
|
else f(prompt(shortText, ""));
|
||||||
|
}
|
||||||
|
function confirmDialog(cm, text, shortText, fs) {
|
||||||
|
if (cm.openConfirm) cm.openConfirm(text, fs);
|
||||||
|
else if (confirm(shortText)) fs[0]();
|
||||||
|
}
|
||||||
|
function parseQuery(query) {
|
||||||
|
var isRE = query.match(/^\/(.*)\/([a-z]*)$/);
|
||||||
|
return isRE ? new RegExp(isRE[1], isRE[2].indexOf("i") == -1 ? "" : "i") : query;
|
||||||
|
}
|
||||||
|
var queryDialog =
|
||||||
|
'Search: <input type="text" style="width: 10em"/> <span style="color: #888">(Use /re/ syntax for regexp search)</span>';
|
||||||
|
function doSearch(cm, rev) {
|
||||||
|
var state = getSearchState(cm);
|
||||||
|
if (state.query) return findNext(cm, rev);
|
||||||
|
dialog(cm, queryDialog, "Search for:", function(query) {
|
||||||
|
cm.operation(function() {
|
||||||
|
if (!query || state.query) return;
|
||||||
|
state.query = parseQuery(query);
|
||||||
|
if (cm.lineCount() < 2000) { // This is too expensive on big documents.
|
||||||
|
for (var cursor = getSearchCursor(cm, state.query); cursor.findNext();)
|
||||||
|
state.marked.push(cm.markText(cursor.from(), cursor.to(), "CodeMirror-searching"));
|
||||||
|
}
|
||||||
|
state.posFrom = state.posTo = cm.getCursor();
|
||||||
|
findNext(cm, rev);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
function findNext(cm, rev) {cm.operation(function() {
|
||||||
|
var state = getSearchState(cm);
|
||||||
|
var cursor = getSearchCursor(cm, state.query, rev ? state.posFrom : state.posTo);
|
||||||
|
if (!cursor.find(rev)) {
|
||||||
|
cursor = getSearchCursor(cm, state.query, rev ? {line: cm.lineCount() - 1} : {line: 0, ch: 0});
|
||||||
|
if (!cursor.find(rev)) return;
|
||||||
|
}
|
||||||
|
cm.setSelection(cursor.from(), cursor.to());
|
||||||
|
state.posFrom = cursor.from(); state.posTo = cursor.to();
|
||||||
|
});}
|
||||||
|
function clearSearch(cm) {cm.operation(function() {
|
||||||
|
var state = getSearchState(cm);
|
||||||
|
if (!state.query) return;
|
||||||
|
state.query = null;
|
||||||
|
for (var i = 0; i < state.marked.length; ++i) state.marked[i].clear();
|
||||||
|
state.marked.length = 0;
|
||||||
|
});}
|
||||||
|
|
||||||
|
var replaceQueryDialog =
|
||||||
|
'Replace: <input type="text" style="width: 10em"/> <span style="color: #888">(Use /re/ syntax for regexp search)</span>';
|
||||||
|
var replacementQueryDialog = 'With: <input type="text" style="width: 10em"/>';
|
||||||
|
var doReplaceConfirm = "Replace? <button>Yes</button> <button>No</button> <button>Stop</button>";
|
||||||
|
function replace(cm, all) {
|
||||||
|
dialog(cm, replaceQueryDialog, "Replace:", function(query) {
|
||||||
|
if (!query) return;
|
||||||
|
query = parseQuery(query);
|
||||||
|
dialog(cm, replacementQueryDialog, "Replace with:", function(text) {
|
||||||
|
if (all) {
|
||||||
|
cm.compoundChange(function() { cm.operation(function() {
|
||||||
|
for (var cursor = getSearchCursor(cm, query); cursor.findNext();) {
|
||||||
|
if (typeof query != "string") {
|
||||||
|
var match = cm.getRange(cursor.from(), cursor.to()).match(query);
|
||||||
|
cursor.replace(text.replace(/\$(\d)/, function(w, i) {return match[i];}));
|
||||||
|
} else cursor.replace(text);
|
||||||
|
}
|
||||||
|
});});
|
||||||
|
} else {
|
||||||
|
clearSearch(cm);
|
||||||
|
var cursor = getSearchCursor(cm, query, cm.getCursor());
|
||||||
|
function advance() {
|
||||||
|
var start = cursor.from(), match;
|
||||||
|
if (!(match = cursor.findNext())) {
|
||||||
|
cursor = getSearchCursor(cm, query);
|
||||||
|
if (!(match = cursor.findNext()) ||
|
||||||
|
(start && cursor.from().line == start.line && cursor.from().ch == start.ch)) return;
|
||||||
|
}
|
||||||
|
cm.setSelection(cursor.from(), cursor.to());
|
||||||
|
confirmDialog(cm, doReplaceConfirm, "Replace?",
|
||||||
|
[function() {doReplace(match);}, advance]);
|
||||||
|
}
|
||||||
|
function doReplace(match) {
|
||||||
|
cursor.replace(typeof query == "string" ? text :
|
||||||
|
text.replace(/\$(\d)/, function(w, i) {return match[i];}));
|
||||||
|
advance();
|
||||||
|
}
|
||||||
|
advance();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.commands.find = function(cm) {clearSearch(cm); doSearch(cm);};
|
||||||
|
CodeMirror.commands.findNext = doSearch;
|
||||||
|
CodeMirror.commands.findPrev = function(cm) {doSearch(cm, true);};
|
||||||
|
CodeMirror.commands.clearSearch = clearSearch;
|
||||||
|
CodeMirror.commands.replace = replace;
|
||||||
|
CodeMirror.commands.replaceAll = function(cm) {replace(cm, true);};
|
||||||
|
})();
|
119
codemirror-2.36/lib/util/searchcursor.js
Normal file
119
codemirror-2.36/lib/util/searchcursor.js
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
(function(){
|
||||||
|
function SearchCursor(cm, query, pos, caseFold) {
|
||||||
|
this.atOccurrence = false; this.cm = cm;
|
||||||
|
if (caseFold == null && typeof query == "string") caseFold = false;
|
||||||
|
|
||||||
|
pos = pos ? cm.clipPos(pos) : {line: 0, ch: 0};
|
||||||
|
this.pos = {from: pos, to: pos};
|
||||||
|
|
||||||
|
// The matches method is filled in based on the type of query.
|
||||||
|
// It takes a position and a direction, and returns an object
|
||||||
|
// describing the next occurrence of the query, or null if no
|
||||||
|
// more matches were found.
|
||||||
|
if (typeof query != "string") { // Regexp match
|
||||||
|
if (!query.global) query = new RegExp(query.source, query.ignoreCase ? "ig" : "g");
|
||||||
|
this.matches = function(reverse, pos) {
|
||||||
|
if (reverse) {
|
||||||
|
query.lastIndex = 0;
|
||||||
|
var line = cm.getLine(pos.line).slice(0, pos.ch), match = query.exec(line), start = 0;
|
||||||
|
while (match) {
|
||||||
|
start += match.index + 1;
|
||||||
|
line = line.slice(start);
|
||||||
|
query.lastIndex = 0;
|
||||||
|
var newmatch = query.exec(line);
|
||||||
|
if (newmatch) match = newmatch;
|
||||||
|
else break;
|
||||||
|
}
|
||||||
|
start--;
|
||||||
|
} else {
|
||||||
|
query.lastIndex = pos.ch;
|
||||||
|
var line = cm.getLine(pos.line), match = query.exec(line),
|
||||||
|
start = match && match.index;
|
||||||
|
}
|
||||||
|
if (match)
|
||||||
|
return {from: {line: pos.line, ch: start},
|
||||||
|
to: {line: pos.line, ch: start + match[0].length},
|
||||||
|
match: match};
|
||||||
|
};
|
||||||
|
} else { // String query
|
||||||
|
if (caseFold) query = query.toLowerCase();
|
||||||
|
var fold = caseFold ? function(str){return str.toLowerCase();} : function(str){return str;};
|
||||||
|
var target = query.split("\n");
|
||||||
|
// Different methods for single-line and multi-line queries
|
||||||
|
if (target.length == 1)
|
||||||
|
this.matches = function(reverse, pos) {
|
||||||
|
var line = fold(cm.getLine(pos.line)), len = query.length, match;
|
||||||
|
if (reverse ? (pos.ch >= len && (match = line.lastIndexOf(query, pos.ch - len)) != -1)
|
||||||
|
: (match = line.indexOf(query, pos.ch)) != -1)
|
||||||
|
return {from: {line: pos.line, ch: match},
|
||||||
|
to: {line: pos.line, ch: match + len}};
|
||||||
|
};
|
||||||
|
else
|
||||||
|
this.matches = function(reverse, pos) {
|
||||||
|
var ln = pos.line, idx = (reverse ? target.length - 1 : 0), match = target[idx], line = fold(cm.getLine(ln));
|
||||||
|
var offsetA = (reverse ? line.indexOf(match) + match.length : line.lastIndexOf(match));
|
||||||
|
if (reverse ? offsetA >= pos.ch || offsetA != match.length
|
||||||
|
: offsetA <= pos.ch || offsetA != line.length - match.length)
|
||||||
|
return;
|
||||||
|
for (;;) {
|
||||||
|
if (reverse ? !ln : ln == cm.lineCount() - 1) return;
|
||||||
|
line = fold(cm.getLine(ln += reverse ? -1 : 1));
|
||||||
|
match = target[reverse ? --idx : ++idx];
|
||||||
|
if (idx > 0 && idx < target.length - 1) {
|
||||||
|
if (line != match) return;
|
||||||
|
else continue;
|
||||||
|
}
|
||||||
|
var offsetB = (reverse ? line.lastIndexOf(match) : line.indexOf(match) + match.length);
|
||||||
|
if (reverse ? offsetB != line.length - match.length : offsetB != match.length)
|
||||||
|
return;
|
||||||
|
var start = {line: pos.line, ch: offsetA}, end = {line: ln, ch: offsetB};
|
||||||
|
return {from: reverse ? end : start, to: reverse ? start : end};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SearchCursor.prototype = {
|
||||||
|
findNext: function() {return this.find(false);},
|
||||||
|
findPrevious: function() {return this.find(true);},
|
||||||
|
|
||||||
|
find: function(reverse) {
|
||||||
|
var self = this, pos = this.cm.clipPos(reverse ? this.pos.from : this.pos.to);
|
||||||
|
function savePosAndFail(line) {
|
||||||
|
var pos = {line: line, ch: 0};
|
||||||
|
self.pos = {from: pos, to: pos};
|
||||||
|
self.atOccurrence = false;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
if (this.pos = this.matches(reverse, pos)) {
|
||||||
|
this.atOccurrence = true;
|
||||||
|
return this.pos.match || true;
|
||||||
|
}
|
||||||
|
if (reverse) {
|
||||||
|
if (!pos.line) return savePosAndFail(0);
|
||||||
|
pos = {line: pos.line-1, ch: this.cm.getLine(pos.line-1).length};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var maxLine = this.cm.lineCount();
|
||||||
|
if (pos.line == maxLine - 1) return savePosAndFail(maxLine);
|
||||||
|
pos = {line: pos.line+1, ch: 0};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
from: function() {if (this.atOccurrence) return this.pos.from;},
|
||||||
|
to: function() {if (this.atOccurrence) return this.pos.to;},
|
||||||
|
|
||||||
|
replace: function(newText) {
|
||||||
|
var self = this;
|
||||||
|
if (this.atOccurrence)
|
||||||
|
self.pos.to = this.cm.replaceRange(newText, self.pos.from, self.pos.to);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
CodeMirror.defineExtension("getSearchCursor", function(query, pos, caseFold) {
|
||||||
|
return new SearchCursor(this, query, pos, caseFold);
|
||||||
|
});
|
||||||
|
})();
|
16
codemirror-2.36/lib/util/simple-hint.css
Normal file
16
codemirror-2.36/lib/util/simple-hint.css
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
.CodeMirror-completions {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 10;
|
||||||
|
overflow: hidden;
|
||||||
|
-webkit-box-shadow: 2px 3px 5px rgba(0,0,0,.2);
|
||||||
|
-moz-box-shadow: 2px 3px 5px rgba(0,0,0,.2);
|
||||||
|
box-shadow: 2px 3px 5px rgba(0,0,0,.2);
|
||||||
|
}
|
||||||
|
.CodeMirror-completions select {
|
||||||
|
background: #fafafa;
|
||||||
|
outline: none;
|
||||||
|
border: none;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
102
codemirror-2.36/lib/util/simple-hint.js
Normal file
102
codemirror-2.36/lib/util/simple-hint.js
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
(function() {
|
||||||
|
CodeMirror.simpleHint = function(editor, getHints, givenOptions) {
|
||||||
|
// Determine effective options based on given values and defaults.
|
||||||
|
var options = {}, defaults = CodeMirror.simpleHint.defaults;
|
||||||
|
for (var opt in defaults)
|
||||||
|
if (defaults.hasOwnProperty(opt))
|
||||||
|
options[opt] = (givenOptions && givenOptions.hasOwnProperty(opt) ? givenOptions : defaults)[opt];
|
||||||
|
|
||||||
|
function collectHints(previousToken) {
|
||||||
|
// We want a single cursor position.
|
||||||
|
if (editor.somethingSelected()) return;
|
||||||
|
|
||||||
|
var tempToken = editor.getTokenAt(editor.getCursor());
|
||||||
|
|
||||||
|
// Don't show completions if token has changed and the option is set.
|
||||||
|
if (options.closeOnTokenChange && previousToken != null &&
|
||||||
|
(tempToken.start != previousToken.start || tempToken.className != previousToken.className)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = getHints(editor);
|
||||||
|
if (!result || !result.list.length) return;
|
||||||
|
var completions = result.list;
|
||||||
|
function insert(str) {
|
||||||
|
editor.replaceRange(str, result.from, result.to);
|
||||||
|
}
|
||||||
|
// When there is only one completion, use it directly.
|
||||||
|
if (options.completeSingle && completions.length == 1) {
|
||||||
|
insert(completions[0]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build the select widget
|
||||||
|
var complete = document.createElement("div");
|
||||||
|
complete.className = "CodeMirror-completions";
|
||||||
|
var sel = complete.appendChild(document.createElement("select"));
|
||||||
|
// Opera doesn't move the selection when pressing up/down in a
|
||||||
|
// multi-select, but it does properly support the size property on
|
||||||
|
// single-selects, so no multi-select is necessary.
|
||||||
|
if (!window.opera) sel.multiple = true;
|
||||||
|
for (var i = 0; i < completions.length; ++i) {
|
||||||
|
var opt = sel.appendChild(document.createElement("option"));
|
||||||
|
opt.appendChild(document.createTextNode(completions[i]));
|
||||||
|
}
|
||||||
|
sel.firstChild.selected = true;
|
||||||
|
sel.size = Math.min(10, completions.length);
|
||||||
|
var pos = options.alignWithWord ? editor.charCoords(result.from) : editor.cursorCoords();
|
||||||
|
complete.style.left = pos.x + "px";
|
||||||
|
complete.style.top = pos.yBot + "px";
|
||||||
|
document.body.appendChild(complete);
|
||||||
|
// If we're at the edge of the screen, then we want the menu to appear on the left of the cursor.
|
||||||
|
var winW = window.innerWidth || Math.max(document.body.offsetWidth, document.documentElement.offsetWidth);
|
||||||
|
if(winW - pos.x < sel.clientWidth)
|
||||||
|
complete.style.left = (pos.x - sel.clientWidth) + "px";
|
||||||
|
// Hack to hide the scrollbar.
|
||||||
|
if (completions.length <= 10)
|
||||||
|
complete.style.width = (sel.clientWidth - 1) + "px";
|
||||||
|
|
||||||
|
var done = false;
|
||||||
|
function close() {
|
||||||
|
if (done) return;
|
||||||
|
done = true;
|
||||||
|
complete.parentNode.removeChild(complete);
|
||||||
|
}
|
||||||
|
function pick() {
|
||||||
|
insert(completions[sel.selectedIndex]);
|
||||||
|
close();
|
||||||
|
setTimeout(function(){editor.focus();}, 50);
|
||||||
|
}
|
||||||
|
CodeMirror.connect(sel, "blur", close);
|
||||||
|
CodeMirror.connect(sel, "keydown", function(event) {
|
||||||
|
var code = event.keyCode;
|
||||||
|
// Enter
|
||||||
|
if (code == 13) {CodeMirror.e_stop(event); pick();}
|
||||||
|
// Escape
|
||||||
|
else if (code == 27) {CodeMirror.e_stop(event); close(); editor.focus();}
|
||||||
|
else if (code != 38 && code != 40 && code != 33 && code != 34 && !CodeMirror.isModifierKey(event)) {
|
||||||
|
close(); editor.focus();
|
||||||
|
// Pass the event to the CodeMirror instance so that it can handle things like backspace properly.
|
||||||
|
editor.triggerOnKeyDown(event);
|
||||||
|
// Don't show completions if the code is backspace and the option is set.
|
||||||
|
if (!options.closeOnBackspace || code != 8) {
|
||||||
|
setTimeout(function(){collectHints(tempToken);}, 50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
CodeMirror.connect(sel, "dblclick", pick);
|
||||||
|
|
||||||
|
sel.focus();
|
||||||
|
// Opera sometimes ignores focusing a freshly created node
|
||||||
|
if (window.opera) setTimeout(function(){if (!done) sel.focus();}, 100);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return collectHints();
|
||||||
|
};
|
||||||
|
CodeMirror.simpleHint.defaults = {
|
||||||
|
closeOnBackspace: true,
|
||||||
|
closeOnTokenChange: false,
|
||||||
|
completeSingle: true,
|
||||||
|
alignWithWord: true
|
||||||
|
};
|
||||||
|
})();
|
131
codemirror-2.36/lib/util/xml-hint.js
Normal file
131
codemirror-2.36/lib/util/xml-hint.js
Normal file
|
@ -0,0 +1,131 @@
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
|
||||||
|
CodeMirror.xmlHints = [];
|
||||||
|
|
||||||
|
CodeMirror.xmlHint = function(cm, simbol) {
|
||||||
|
|
||||||
|
if(simbol.length > 0) {
|
||||||
|
var cursor = cm.getCursor();
|
||||||
|
cm.replaceSelection(simbol);
|
||||||
|
cursor = {line: cursor.line, ch: cursor.ch + 1};
|
||||||
|
cm.setCursor(cursor);
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.simpleHint(cm, getHint);
|
||||||
|
};
|
||||||
|
|
||||||
|
var getHint = function(cm) {
|
||||||
|
|
||||||
|
var cursor = cm.getCursor();
|
||||||
|
|
||||||
|
if (cursor.ch > 0) {
|
||||||
|
|
||||||
|
var text = cm.getRange({line: 0, ch: 0}, cursor);
|
||||||
|
var typed = '';
|
||||||
|
var simbol = '';
|
||||||
|
for(var i = text.length - 1; i >= 0; i--) {
|
||||||
|
if(text[i] == ' ' || text[i] == '<') {
|
||||||
|
simbol = text[i];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
typed = text[i] + typed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
text = text.slice(0, text.length - typed.length);
|
||||||
|
|
||||||
|
var path = getActiveElement(cm, text) + simbol;
|
||||||
|
var hints = CodeMirror.xmlHints[path];
|
||||||
|
|
||||||
|
if(typeof hints === 'undefined')
|
||||||
|
hints = [''];
|
||||||
|
else {
|
||||||
|
hints = hints.slice(0);
|
||||||
|
for (var i = hints.length - 1; i >= 0; i--) {
|
||||||
|
if(hints[i].indexOf(typed) != 0)
|
||||||
|
hints.splice(i, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
list: hints,
|
||||||
|
from: { line: cursor.line, ch: cursor.ch - typed.length },
|
||||||
|
to: cursor
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
var getActiveElement = function(codeMirror, text) {
|
||||||
|
|
||||||
|
var element = '';
|
||||||
|
|
||||||
|
if(text.length >= 0) {
|
||||||
|
|
||||||
|
var regex = new RegExp('<([^!?][^\\s/>]*).*?>', 'g');
|
||||||
|
|
||||||
|
var matches = [];
|
||||||
|
var match;
|
||||||
|
while ((match = regex.exec(text)) != null) {
|
||||||
|
matches.push({
|
||||||
|
tag: match[1],
|
||||||
|
selfclose: (match[0].slice(match[0].length - 2) === '/>')
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = matches.length - 1, skip = 0; i >= 0; i--) {
|
||||||
|
|
||||||
|
var item = matches[i];
|
||||||
|
|
||||||
|
if (item.tag[0] == '/')
|
||||||
|
{
|
||||||
|
skip++;
|
||||||
|
}
|
||||||
|
else if (item.selfclose == false)
|
||||||
|
{
|
||||||
|
if (skip > 0)
|
||||||
|
{
|
||||||
|
skip--;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
element = '<' + item.tag + '>' + element;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
element += getOpenTag(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
return element;
|
||||||
|
};
|
||||||
|
|
||||||
|
var getOpenTag = function(text) {
|
||||||
|
|
||||||
|
var open = text.lastIndexOf('<');
|
||||||
|
var close = text.lastIndexOf('>');
|
||||||
|
|
||||||
|
if (close < open)
|
||||||
|
{
|
||||||
|
text = text.slice(open);
|
||||||
|
|
||||||
|
if(text != '<') {
|
||||||
|
|
||||||
|
var space = text.indexOf(' ');
|
||||||
|
if(space < 0)
|
||||||
|
space = text.indexOf('\t');
|
||||||
|
if(space < 0)
|
||||||
|
space = text.indexOf('\n');
|
||||||
|
|
||||||
|
if (space < 0)
|
||||||
|
space = text.length;
|
||||||
|
|
||||||
|
return text.slice(0, space);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
};
|
||||||
|
|
||||||
|
})();
|
285
codemirror-2.36/mode/clike/clike.js
Normal file
285
codemirror-2.36/mode/clike/clike.js
Normal file
|
@ -0,0 +1,285 @@
|
||||||
|
CodeMirror.defineMode("clike", function(config, parserConfig) {
|
||||||
|
var indentUnit = config.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 == "'") {
|
||||||
|
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();
|
||||||
|
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 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") && curPunc != ';') || (ctx.type == "statement" && curPunc == "newstatement"))
|
||||||
|
pushContext(state, stream.column(), "statement");
|
||||||
|
state.startOfLine = false;
|
||||||
|
return style;
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
if (state.tokenize == tokenComment) return CodeMirror.Pass;
|
||||||
|
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: "{}"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
function words(str) {
|
||||||
|
var obj = {}, words = str.split(" ");
|
||||||
|
for (var i = 0; i < words.length; ++i) obj[words[i]] = true;
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
var cKeywords = "auto if break int case long char register continue return default short do sizeof " +
|
||||||
|
"double static else struct entry switch extern typedef float union for unsigned " +
|
||||||
|
"goto while enum void const signed volatile";
|
||||||
|
|
||||||
|
function cppHook(stream, state) {
|
||||||
|
if (!state.startOfLine) return false;
|
||||||
|
stream.skipToEnd();
|
||||||
|
return "meta";
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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";
|
||||||
|
}
|
||||||
|
|
||||||
|
function mimes(ms, mode) {
|
||||||
|
for (var i = 0; i < ms.length; ++i) CodeMirror.defineMIME(ms[i], mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
mimes(["text/x-csrc", "text/x-c", "text/x-chdr"], {
|
||||||
|
name: "clike",
|
||||||
|
keywords: words(cKeywords),
|
||||||
|
blockKeywords: words("case do else for if switch while struct"),
|
||||||
|
atoms: words("null"),
|
||||||
|
hooks: {"#": cppHook}
|
||||||
|
});
|
||||||
|
mimes(["text/x-c++src", "text/x-c++hdr"], {
|
||||||
|
name: "clike",
|
||||||
|
keywords: words(cKeywords + " asm dynamic_cast namespace reinterpret_cast try bool explicit new " +
|
||||||
|
"static_cast typeid catch operator template typename class friend private " +
|
||||||
|
"this using const_cast inline public throw virtual delete mutable protected " +
|
||||||
|
"wchar_t"),
|
||||||
|
blockKeywords: words("catch class do else finally for if struct switch try while"),
|
||||||
|
atoms: words("true false null"),
|
||||||
|
hooks: {"#": cppHook}
|
||||||
|
});
|
||||||
|
CodeMirror.defineMIME("text/x-java", {
|
||||||
|
name: "clike",
|
||||||
|
keywords: words("abstract assert boolean break byte case catch char class const continue default " +
|
||||||
|
"do double else enum extends final finally float for goto if implements import " +
|
||||||
|
"instanceof int interface long native new package private protected public " +
|
||||||
|
"return short static strictfp super switch synchronized this throw throws transient " +
|
||||||
|
"try void volatile while"),
|
||||||
|
blockKeywords: words("catch class do else finally for if switch try while"),
|
||||||
|
atoms: words("true false null"),
|
||||||
|
hooks: {
|
||||||
|
"@": function(stream, state) {
|
||||||
|
stream.eatWhile(/[\w\$_]/);
|
||||||
|
return "meta";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
CodeMirror.defineMIME("text/x-csharp", {
|
||||||
|
name: "clike",
|
||||||
|
keywords: words("abstract as 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"),
|
||||||
|
blockKeywords: words("catch class do else finally for foreach if struct switch try while"),
|
||||||
|
builtin: words("Boolean Byte Char DateTime DateTimeOffset Decimal Double" +
|
||||||
|
" Guid Int16 Int32 Int64 Object SByte Single String TimeSpan UInt16 UInt32" +
|
||||||
|
" UInt64 bool byte char decimal double short int long object" +
|
||||||
|
" sbyte float string ushort uint ulong"),
|
||||||
|
atoms: words("true false null"),
|
||||||
|
hooks: {
|
||||||
|
"@": function(stream, state) {
|
||||||
|
if (stream.eat('"')) {
|
||||||
|
state.tokenize = tokenAtString;
|
||||||
|
return tokenAtString(stream, state);
|
||||||
|
}
|
||||||
|
stream.eatWhile(/[\w\$_]/);
|
||||||
|
return "meta";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
CodeMirror.defineMIME("text/x-scala", {
|
||||||
|
name: "clike",
|
||||||
|
keywords: words(
|
||||||
|
|
||||||
|
/* scala */
|
||||||
|
"abstract case catch class def do else extends false final finally for forSome if " +
|
||||||
|
"implicit import lazy match new null object override package private protected return " +
|
||||||
|
"sealed super this throw trait try trye type val var while with yield _ : = => <- <: " +
|
||||||
|
"<% >: # @ " +
|
||||||
|
|
||||||
|
/* package scala */
|
||||||
|
"assert assume require print println printf readLine readBoolean readByte readShort " +
|
||||||
|
"readChar readInt readLong readFloat readDouble " +
|
||||||
|
|
||||||
|
"AnyVal App Application Array BufferedIterator BigDecimal BigInt Char Console Either " +
|
||||||
|
"Enumeration Equiv Error Exception Fractional Function IndexedSeq 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"
|
||||||
|
|
||||||
|
|
||||||
|
),
|
||||||
|
blockKeywords: words("catch class do else finally for forSome if match switch try while"),
|
||||||
|
atoms: words("true false null"),
|
||||||
|
hooks: {
|
||||||
|
"@": function(stream, state) {
|
||||||
|
stream.eatWhile(/[\w\$_]/);
|
||||||
|
return "meta";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}());
|
102
codemirror-2.36/mode/clike/index.html
Normal file
102
codemirror-2.36/mode/clike/index.html
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: C-like mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="clike.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>.CodeMirror {border: 2px inset #dee;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: C-like mode</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="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></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
mode: "text/x-csrc"
|
||||||
|
});
|
||||||
|
</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), <code>text/x-c++src</code> (C++
|
||||||
|
code), <code>text/x-java</code> (Java
|
||||||
|
code), <code>text/x-csharp</code> (C#).</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
766
codemirror-2.36/mode/clike/scala.html
Normal file
766
codemirror-2.36/mode/clike/scala.html
Normal file
|
@ -0,0 +1,766 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: C-like mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<link rel="stylesheet" href="../../theme/ambiance.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="clike.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>
|
||||||
|
body
|
||||||
|
{
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
max-width:inherit;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
html, form, .CodeMirror, .CodeMirror-scroll
|
||||||
|
{
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<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>
|
||||||
|
</body>
|
||||||
|
</html>
|
206
codemirror-2.36/mode/clojure/clojure.js
Normal file
206
codemirror-2.36/mode/clojure/clojure.js
Normal file
|
@ -0,0 +1,206 @@
|
||||||
|
/**
|
||||||
|
* Author: Hans Engel
|
||||||
|
* Branched from CodeMirror's Scheme mode (by Koh Zi Han, based on implementation by Koh Zi Chun)
|
||||||
|
*/
|
||||||
|
CodeMirror.defineMode("clojure", function (config, mode) {
|
||||||
|
var BUILTIN = "builtin", COMMENT = "comment", STRING = "string", TAG = "tag",
|
||||||
|
ATOM = "atom", NUMBER = "number", BRACKET = "bracket", KEYWORD = "keyword";
|
||||||
|
var INDENT_WORD_SKIP = 2, KEYWORDS_SKIP = 1;
|
||||||
|
|
||||||
|
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* *e *err* *file* *flush-on-newline* *in* *macro-meta* *math-context* *ns* *out* *print-dup* *print-length* *print-level* *print-meta* *print-readably* *read-eval* *source-path* *use-context-classloader* *warn-on-reflection* + - / < <= = == > >= accessor aclone agent 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 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* butlast byte byte-array bytes case 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 concat cond condp conj conj! cons constantly construct-proxy contains? count counted? create-ns create-struct cycle dec decimal? declare definline defmacro defmethod defmulti defn defn- defonce defstruct delay delay? deliver deref derive descendants destructure disj disj! dissoc dissoc! distinct distinct? doall doc dorun doseq dosync dotimes doto double double-array doubles drop drop-last drop-while empty empty? ensure enumeration-seq eval even? every? extend extend-protocol extend-type extends? extenders false? ffirst file-seq filter find find-doc find-ns find-var first float float-array float? floats flush fn fn? fnext for force format 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 hash hash-map hash-set identical? identity if-let if-not ifn? import in-ns inc init-proxy instance? int int-array integer? interleave intern interpose into into-array ints io! isa? iterate iterator-seq juxt 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? mapcat max max-key memfn memoize merge merge-with meta method-sig methods min min-key mod name namespace 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 num number? odd? or parents partial partition pcalls peek persistent! pmap pop pop! pop-thread-bindings pos? pr pr-str prefer-method prefers primitives-classnames print print-ctor print-doc print-dup print-method print-namespace-doc print-simple print-special-doc 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 range ratio? rational? rationalize re-find re-groups re-matcher re-matches re-pattern re-seq read read-line read-string reify reduce ref ref-history-count ref-max-history ref-min-history ref-set refer refer-clojure release-pending-sends rem remove remove-method remove-ns repeat repeatedly replace replicate require reset! reset-meta! resolve rest resultset-seq reverse reversible? rseq rsubseq satisfies? second select-keys send send-off seq seq? seque sequence sequential? set set-validator! set? short short-array shorts shutdown-agents slurp some sort sort-by sorted-map sorted-map-by sorted-set sorted-set-by sorted? special-form-anchor special-symbol? split-at split-with str stream? string? struct struct-map subs subseq subvec supers swap! symbol symbol? sync syntax-symbol-anchor take take-last take-nth take-while test the-ns time to-array to-array-2d trampoline transient tree-seq true? type unchecked-add unchecked-dec unchecked-divide unchecked-inc unchecked-multiply unchecked-negate unchecked-remainder unchecked-subtract underive unquote unquote-splicing update-in update-proxy use val vals var-get var-set var? vary-meta vec vector vector? 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 xml-seq");
|
||||||
|
|
||||||
|
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\(\[\;\)\]]/,
|
||||||
|
basic: /[\w\$_\-]/,
|
||||||
|
lang_keyword: /[\w*+!\-_?:\/]/
|
||||||
|
};
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
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 = stream.indentation();
|
||||||
|
}
|
||||||
|
|
||||||
|
// skip spaces
|
||||||
|
if (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 == "'" && !( 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 == "[") {
|
||||||
|
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) ||
|
||||||
|
/^(?:def|with)/.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 1 space after
|
||||||
|
pushStack(state, indentTemp + 1, 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 == "]") {
|
||||||
|
returnType = BRACKET;
|
||||||
|
if (state.indentStack != null && state.indentStack.type == (ch == ")" ? "(" : "[")) {
|
||||||
|
popStack(state);
|
||||||
|
}
|
||||||
|
} else if ( ch == ":" ) {
|
||||||
|
stream.eatWhile(tests.lang_keyword);
|
||||||
|
return ATOM;
|
||||||
|
} else {
|
||||||
|
stream.eatWhile(tests.basic);
|
||||||
|
|
||||||
|
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 = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return returnType;
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function (state, textAfter) {
|
||||||
|
if (state.indentStack == null) return state.indentation;
|
||||||
|
return state.indentStack.indent;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-clojure", "clojure");
|
67
codemirror-2.36/mode/clojure/index.html
Normal file
67
codemirror-2.36/mode/clojure/index.html
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Clojure mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="clojure.js"></script>
|
||||||
|
<style>.CodeMirror {background: #f8f8f8;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Clojure mode</h1>
|
||||||
|
<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) </textarea></form>
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/x-clojure</code>.</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
22
codemirror-2.36/mode/coffeescript/LICENSE
Normal file
22
codemirror-2.36/mode/coffeescript/LICENSE
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
The MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2011 Jeff Pickhardt
|
||||||
|
Modified from the Python CodeMirror mode, Copyright (c) 2010 Timothy Farrell
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
346
codemirror-2.36/mode/coffeescript/coffeescript.js
Normal file
346
codemirror-2.36/mode/coffeescript/coffeescript.js
Normal file
|
@ -0,0 +1,346 @@
|
||||||
|
/**
|
||||||
|
* Link to the project's GitHub page:
|
||||||
|
* https://github.com/pickhardt/coffeescript-codemirror-mode
|
||||||
|
*/
|
||||||
|
CodeMirror.defineMode('coffeescript', function(conf) {
|
||||||
|
var ERRORCLASS = 'error';
|
||||||
|
|
||||||
|
function wordRegexp(words) {
|
||||||
|
return new RegExp("^((" + words.join(")|(") + "))\\b");
|
||||||
|
}
|
||||||
|
|
||||||
|
var singleOperators = new RegExp("^[\\+\\-\\*/%&|\\^~<>!\?]");
|
||||||
|
var singleDelimiters = new RegExp('^[\\(\\)\\[\\]\\{\\},:`=;\\.]');
|
||||||
|
var doubleOperators = new RegExp("^((\->)|(\=>)|(\\+\\+)|(\\+\\=)|(\\-\\-)|(\\-\\=)|(\\*\\*)|(\\*\\=)|(\\/\\/)|(\\/\\=)|(==)|(!=)|(<=)|(>=)|(<>)|(<<)|(>>)|(//))");
|
||||||
|
var doubleDelimiters = new RegExp("^((\\.\\.)|(\\+=)|(\\-=)|(\\*=)|(%=)|(/=)|(&=)|(\\|=)|(\\^=))");
|
||||||
|
var tripleDelimiters = new RegExp("^((\\.\\.\\.)|(//=)|(>>=)|(<<=)|(\\*\\*=))");
|
||||||
|
var identifiers = new RegExp("^[_A-Za-z$][_A-Za-z$0-9]*");
|
||||||
|
var properties = new RegExp("^(@|this\.)[_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'];
|
||||||
|
|
||||||
|
var keywords = wordRegexp(indentKeywords.concat(commonKeywords));
|
||||||
|
|
||||||
|
indentKeywords = wordRegexp(indentKeywords);
|
||||||
|
|
||||||
|
|
||||||
|
var stringPrefixes = new RegExp("^('{3}|\"{3}|['\"])");
|
||||||
|
var regexPrefixes = new RegExp("^(/{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()) {
|
||||||
|
var scopeOffset = state.scopes[0].offset;
|
||||||
|
if (stream.eatSpace()) {
|
||||||
|
var lineOffset = stream.indentation();
|
||||||
|
if (lineOffset > scopeOffset) {
|
||||||
|
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(), '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(), 'string-2');
|
||||||
|
return state.tokenize(stream, state);
|
||||||
|
} else {
|
||||||
|
stream.backUp(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle operators and delimiters
|
||||||
|
if (stream.match(tripleDelimiters) || stream.match(doubleDelimiters)) {
|
||||||
|
return 'punctuation';
|
||||||
|
}
|
||||||
|
if (stream.match(doubleOperators)
|
||||||
|
|| stream.match(singleOperators)
|
||||||
|
|| stream.match(wordOperators)) {
|
||||||
|
return 'operator';
|
||||||
|
}
|
||||||
|
if (stream.match(singleDelimiters)) {
|
||||||
|
return 'punctuation';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stream.match(constants)) {
|
||||||
|
return 'atom';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stream.match(keywords)) {
|
||||||
|
return 'keyword';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stream.match(identifiers)) {
|
||||||
|
return 'variable';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stream.match(properties)) {
|
||||||
|
return 'property';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle non-detected items
|
||||||
|
stream.next();
|
||||||
|
return ERRORCLASS;
|
||||||
|
}
|
||||||
|
|
||||||
|
function tokenFactory(delimiter, outclass) {
|
||||||
|
var singleline = delimiter.length == 1;
|
||||||
|
return function tokenString(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 (conf.mode.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 indentUnit = 0;
|
||||||
|
if (type === 'coffee') {
|
||||||
|
for (var i = 0; i < state.scopes.length; i++) {
|
||||||
|
if (state.scopes[i].type === 'coffee') {
|
||||||
|
indentUnit = state.scopes[i].offset + conf.indentUnit;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
indentUnit = stream.column() + stream.current().length;
|
||||||
|
}
|
||||||
|
state.scopes.unshift({
|
||||||
|
offset: indentUnit,
|
||||||
|
type: type
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function dedent(stream, state) {
|
||||||
|
if (state.scopes.length == 1) return;
|
||||||
|
if (state.scopes[0].type === 'coffee') {
|
||||||
|
var _indent = stream.indentation();
|
||||||
|
var _indent_index = -1;
|
||||||
|
for (var i = 0; i < state.scopes.length; ++i) {
|
||||||
|
if (_indent === state.scopes[i].offset) {
|
||||||
|
_indent_index = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (_indent_index === -1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
while (state.scopes[0].offset !== _indent) {
|
||||||
|
state.scopes.shift();
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
state.scopes.shift();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function tokenLexer(stream, state) {
|
||||||
|
var style = state.tokenize(stream, state);
|
||||||
|
var current = stream.current();
|
||||||
|
|
||||||
|
// Handle '.' connected identifiers
|
||||||
|
if (current === '.') {
|
||||||
|
style = state.tokenize(stream, state);
|
||||||
|
current = stream.current();
|
||||||
|
if (style === 'variable') {
|
||||||
|
return 'variable';
|
||||||
|
} else {
|
||||||
|
return ERRORCLASS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle scope changes.
|
||||||
|
if (current === 'return') {
|
||||||
|
state.dedent += 1;
|
||||||
|
}
|
||||||
|
if (((current === '->' || current === '=>') &&
|
||||||
|
!state.lambda &&
|
||||||
|
state.scopes[0].type == 'coffee' &&
|
||||||
|
stream.peek() === '')
|
||||||
|
|| 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) {
|
||||||
|
if (dedent(stream, state)) {
|
||||||
|
return ERRORCLASS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (state.dedent > 0 && stream.eol() && state.scopes[0].type == 'coffee') {
|
||||||
|
if (state.scopes.length > 1) state.scopes.shift();
|
||||||
|
state.dedent -= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
|
||||||
|
var external = {
|
||||||
|
startState: function(basecolumn) {
|
||||||
|
return {
|
||||||
|
tokenize: tokenBase,
|
||||||
|
scopes: [{offset:basecolumn || 0, type:'coffee'}],
|
||||||
|
lastToken: null,
|
||||||
|
lambda: false,
|
||||||
|
dedent: 0
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
token: function(stream, state) {
|
||||||
|
var style = tokenLexer(stream, state);
|
||||||
|
|
||||||
|
state.lastToken = {style:style, content: stream.current()};
|
||||||
|
|
||||||
|
if (stream.eol() && stream.lambda) {
|
||||||
|
state.lambda = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return style;
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
if (state.tokenize != tokenBase) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return state.scopes[0].offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
return external;
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME('text/x-coffeescript', 'coffeescript');
|
728
codemirror-2.36/mode/coffeescript/index.html
Normal file
728
codemirror-2.36/mode/coffeescript/index.html
Normal file
|
@ -0,0 +1,728 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: CoffeeScript mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="coffeescript.js"></script>
|
||||||
|
<style>.CodeMirror {border-top: 1px solid silver; border-bottom: 1px solid silver;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: CoffeeScript mode</h1>
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
# CoffeeScript mode for CodeMirror
|
||||||
|
# Copyright (c) 2011 Jeff Pickhardt, released under
|
||||||
|
# the MIT License.
|
||||||
|
#
|
||||||
|
# Modified from the Python CodeMirror mode, which also is
|
||||||
|
# under the MIT License Copyright (c) 2010 Timothy Farrell.
|
||||||
|
#
|
||||||
|
# The following script, Underscore.coffee, is used to
|
||||||
|
# demonstrate CoffeeScript mode for CodeMirror.
|
||||||
|
#
|
||||||
|
# To download CoffeeScript mode for CodeMirror, go to:
|
||||||
|
# https://github.com/pickhardt/coffeescript-codemirror-mode
|
||||||
|
|
||||||
|
# **Underscore.coffee
|
||||||
|
# (c) 2011 Jeremy Ashkenas, DocumentCloud Inc.**
|
||||||
|
# Underscore is freely distributable under the terms of the
|
||||||
|
# [MIT license](http://en.wikipedia.org/wiki/MIT_License).
|
||||||
|
# Portions of Underscore are inspired by or borrowed from
|
||||||
|
# [Prototype.js](http://prototypejs.org/api), Oliver Steele's
|
||||||
|
# [Functional](http://osteele.com), and John Resig's
|
||||||
|
# [Micro-Templating](http://ejohn.org).
|
||||||
|
# For all details and documentation:
|
||||||
|
# http://documentcloud.github.com/underscore/
|
||||||
|
|
||||||
|
|
||||||
|
# Baseline setup
|
||||||
|
# --------------
|
||||||
|
|
||||||
|
# Establish the root object, `window` in the browser, or `global` on the server.
|
||||||
|
root = this
|
||||||
|
|
||||||
|
|
||||||
|
# Save the previous value of the `_` variable.
|
||||||
|
previousUnderscore = root._
|
||||||
|
|
||||||
|
### Multiline
|
||||||
|
comment
|
||||||
|
###
|
||||||
|
|
||||||
|
# Establish the object that gets thrown to break out of a loop iteration.
|
||||||
|
# `StopIteration` is SOP on Mozilla.
|
||||||
|
breaker = if typeof(StopIteration) is 'undefined' then '__break__' else StopIteration
|
||||||
|
|
||||||
|
|
||||||
|
#### Docco style single line comment (title)
|
||||||
|
|
||||||
|
|
||||||
|
# Helper function to escape **RegExp** contents, because JS doesn't have one.
|
||||||
|
escapeRegExp = (string) -> string.replace(/([.*+?^${}()|[\]\/\\])/g, '\\$1')
|
||||||
|
|
||||||
|
|
||||||
|
# Save bytes in the minified (but not gzipped) version:
|
||||||
|
ArrayProto = Array.prototype
|
||||||
|
ObjProto = Object.prototype
|
||||||
|
|
||||||
|
|
||||||
|
# Create quick reference variables for speed access to core prototypes.
|
||||||
|
slice = ArrayProto.slice
|
||||||
|
unshift = ArrayProto.unshift
|
||||||
|
toString = ObjProto.toString
|
||||||
|
hasOwnProperty = ObjProto.hasOwnProperty
|
||||||
|
propertyIsEnumerable = ObjProto.propertyIsEnumerable
|
||||||
|
|
||||||
|
|
||||||
|
# All **ECMA5** native implementations we hope to use are declared here.
|
||||||
|
nativeForEach = ArrayProto.forEach
|
||||||
|
nativeMap = ArrayProto.map
|
||||||
|
nativeReduce = ArrayProto.reduce
|
||||||
|
nativeReduceRight = ArrayProto.reduceRight
|
||||||
|
nativeFilter = ArrayProto.filter
|
||||||
|
nativeEvery = ArrayProto.every
|
||||||
|
nativeSome = ArrayProto.some
|
||||||
|
nativeIndexOf = ArrayProto.indexOf
|
||||||
|
nativeLastIndexOf = ArrayProto.lastIndexOf
|
||||||
|
nativeIsArray = Array.isArray
|
||||||
|
nativeKeys = Object.keys
|
||||||
|
|
||||||
|
|
||||||
|
# Create a safe reference to the Underscore object for use below.
|
||||||
|
_ = (obj) -> new wrapper(obj)
|
||||||
|
|
||||||
|
|
||||||
|
# Export the Underscore object for **CommonJS**.
|
||||||
|
if typeof(exports) != 'undefined' then exports._ = _
|
||||||
|
|
||||||
|
|
||||||
|
# Export Underscore to global scope.
|
||||||
|
root._ = _
|
||||||
|
|
||||||
|
|
||||||
|
# Current version.
|
||||||
|
_.VERSION = '1.1.0'
|
||||||
|
|
||||||
|
|
||||||
|
# Collection Functions
|
||||||
|
# --------------------
|
||||||
|
|
||||||
|
# The cornerstone, an **each** implementation.
|
||||||
|
# Handles objects implementing **forEach**, arrays, and raw objects.
|
||||||
|
_.each = (obj, iterator, context) ->
|
||||||
|
try
|
||||||
|
if nativeForEach and obj.forEach is nativeForEach
|
||||||
|
obj.forEach iterator, context
|
||||||
|
else if _.isNumber obj.length
|
||||||
|
iterator.call context, obj[i], i, obj for i in [0...obj.length]
|
||||||
|
else
|
||||||
|
iterator.call context, val, key, obj for own key, val of obj
|
||||||
|
catch e
|
||||||
|
throw e if e isnt breaker
|
||||||
|
obj
|
||||||
|
|
||||||
|
|
||||||
|
# Return the results of applying the iterator to each element. Use JavaScript
|
||||||
|
# 1.6's version of **map**, if possible.
|
||||||
|
_.map = (obj, iterator, context) ->
|
||||||
|
return obj.map(iterator, context) if nativeMap and obj.map is nativeMap
|
||||||
|
results = []
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
results.push iterator.call context, value, index, list
|
||||||
|
results
|
||||||
|
|
||||||
|
|
||||||
|
# **Reduce** builds up a single result from a list of values. Also known as
|
||||||
|
# **inject**, or **foldl**. Uses JavaScript 1.8's version of **reduce**, if possible.
|
||||||
|
_.reduce = (obj, iterator, memo, context) ->
|
||||||
|
if nativeReduce and obj.reduce is nativeReduce
|
||||||
|
iterator = _.bind iterator, context if context
|
||||||
|
return obj.reduce iterator, memo
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
memo = iterator.call context, memo, value, index, list
|
||||||
|
memo
|
||||||
|
|
||||||
|
|
||||||
|
# The right-associative version of **reduce**, also known as **foldr**. Uses
|
||||||
|
# JavaScript 1.8's version of **reduceRight**, if available.
|
||||||
|
_.reduceRight = (obj, iterator, memo, context) ->
|
||||||
|
if nativeReduceRight and obj.reduceRight is nativeReduceRight
|
||||||
|
iterator = _.bind iterator, context if context
|
||||||
|
return obj.reduceRight iterator, memo
|
||||||
|
reversed = _.clone(_.toArray(obj)).reverse()
|
||||||
|
_.reduce reversed, iterator, memo, context
|
||||||
|
|
||||||
|
|
||||||
|
# Return the first value which passes a truth test.
|
||||||
|
_.detect = (obj, iterator, context) ->
|
||||||
|
result = null
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
if iterator.call context, value, index, list
|
||||||
|
result = value
|
||||||
|
_.breakLoop()
|
||||||
|
result
|
||||||
|
|
||||||
|
|
||||||
|
# Return all the elements that pass a truth test. Use JavaScript 1.6's
|
||||||
|
# **filter**, if it exists.
|
||||||
|
_.filter = (obj, iterator, context) ->
|
||||||
|
return obj.filter iterator, context if nativeFilter and obj.filter is nativeFilter
|
||||||
|
results = []
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
results.push value if iterator.call context, value, index, list
|
||||||
|
results
|
||||||
|
|
||||||
|
|
||||||
|
# Return all the elements for which a truth test fails.
|
||||||
|
_.reject = (obj, iterator, context) ->
|
||||||
|
results = []
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
results.push value if not iterator.call context, value, index, list
|
||||||
|
results
|
||||||
|
|
||||||
|
|
||||||
|
# Determine whether all of the elements match a truth test. Delegate to
|
||||||
|
# JavaScript 1.6's **every**, if it is present.
|
||||||
|
_.every = (obj, iterator, context) ->
|
||||||
|
iterator ||= _.identity
|
||||||
|
return obj.every iterator, context if nativeEvery and obj.every is nativeEvery
|
||||||
|
result = true
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
_.breakLoop() unless (result = result and iterator.call(context, value, index, list))
|
||||||
|
result
|
||||||
|
|
||||||
|
|
||||||
|
# Determine if at least one element in the object matches a truth test. Use
|
||||||
|
# JavaScript 1.6's **some**, if it exists.
|
||||||
|
_.some = (obj, iterator, context) ->
|
||||||
|
iterator ||= _.identity
|
||||||
|
return obj.some iterator, context if nativeSome and obj.some is nativeSome
|
||||||
|
result = false
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
_.breakLoop() if (result = iterator.call(context, value, index, list))
|
||||||
|
result
|
||||||
|
|
||||||
|
|
||||||
|
# Determine if a given value is included in the array or object,
|
||||||
|
# based on `===`.
|
||||||
|
_.include = (obj, target) ->
|
||||||
|
return _.indexOf(obj, target) isnt -1 if nativeIndexOf and obj.indexOf is nativeIndexOf
|
||||||
|
return true for own key, val of obj when val is target
|
||||||
|
false
|
||||||
|
|
||||||
|
|
||||||
|
# Invoke a method with arguments on every item in a collection.
|
||||||
|
_.invoke = (obj, method) ->
|
||||||
|
args = _.rest arguments, 2
|
||||||
|
(if method then val[method] else val).apply(val, args) for val in obj
|
||||||
|
|
||||||
|
|
||||||
|
# Convenience version of a common use case of **map**: fetching a property.
|
||||||
|
_.pluck = (obj, key) ->
|
||||||
|
_.map(obj, (val) -> val[key])
|
||||||
|
|
||||||
|
|
||||||
|
# Return the maximum item or (item-based computation).
|
||||||
|
_.max = (obj, iterator, context) ->
|
||||||
|
return Math.max.apply(Math, obj) if not iterator and _.isArray(obj)
|
||||||
|
result = computed: -Infinity
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
computed = if iterator then iterator.call(context, value, index, list) else value
|
||||||
|
computed >= result.computed and (result = {value: value, computed: computed})
|
||||||
|
result.value
|
||||||
|
|
||||||
|
|
||||||
|
# Return the minimum element (or element-based computation).
|
||||||
|
_.min = (obj, iterator, context) ->
|
||||||
|
return Math.min.apply(Math, obj) if not iterator and _.isArray(obj)
|
||||||
|
result = computed: Infinity
|
||||||
|
_.each obj, (value, index, list) ->
|
||||||
|
computed = if iterator then iterator.call(context, value, index, list) else value
|
||||||
|
computed < result.computed and (result = {value: value, computed: computed})
|
||||||
|
result.value
|
||||||
|
|
||||||
|
|
||||||
|
# Sort the object's values by a criterion produced by an iterator.
|
||||||
|
_.sortBy = (obj, iterator, context) ->
|
||||||
|
_.pluck(((_.map obj, (value, index, list) ->
|
||||||
|
{value: value, criteria: iterator.call(context, value, index, list)}
|
||||||
|
).sort((left, right) ->
|
||||||
|
a = left.criteria; b = right.criteria
|
||||||
|
if a < b then -1 else if a > b then 1 else 0
|
||||||
|
)), 'value')
|
||||||
|
|
||||||
|
|
||||||
|
# Use a comparator function to figure out at what index an object should
|
||||||
|
# be inserted so as to maintain order. Uses binary search.
|
||||||
|
_.sortedIndex = (array, obj, iterator) ->
|
||||||
|
iterator ||= _.identity
|
||||||
|
low = 0
|
||||||
|
high = array.length
|
||||||
|
while low < high
|
||||||
|
mid = (low + high) >> 1
|
||||||
|
if iterator(array[mid]) < iterator(obj) then low = mid + 1 else high = mid
|
||||||
|
low
|
||||||
|
|
||||||
|
|
||||||
|
# Convert anything iterable into a real, live array.
|
||||||
|
_.toArray = (iterable) ->
|
||||||
|
return [] if (!iterable)
|
||||||
|
return iterable.toArray() if (iterable.toArray)
|
||||||
|
return iterable if (_.isArray(iterable))
|
||||||
|
return slice.call(iterable) if (_.isArguments(iterable))
|
||||||
|
_.values(iterable)
|
||||||
|
|
||||||
|
|
||||||
|
# Return the number of elements in an object.
|
||||||
|
_.size = (obj) -> _.toArray(obj).length
|
||||||
|
|
||||||
|
|
||||||
|
# Array Functions
|
||||||
|
# ---------------
|
||||||
|
|
||||||
|
# Get the first element of an array. Passing `n` will return the first N
|
||||||
|
# values in the array. Aliased as **head**. The `guard` check allows it to work
|
||||||
|
# with **map**.
|
||||||
|
_.first = (array, n, guard) ->
|
||||||
|
if n and not guard then slice.call(array, 0, n) else array[0]
|
||||||
|
|
||||||
|
|
||||||
|
# Returns everything but the first entry of the array. Aliased as **tail**.
|
||||||
|
# Especially useful on the arguments object. Passing an `index` will return
|
||||||
|
# the rest of the values in the array from that index onward. The `guard`
|
||||||
|
# check allows it to work with **map**.
|
||||||
|
_.rest = (array, index, guard) ->
|
||||||
|
slice.call(array, if _.isUndefined(index) or guard then 1 else index)
|
||||||
|
|
||||||
|
|
||||||
|
# Get the last element of an array.
|
||||||
|
_.last = (array) -> array[array.length - 1]
|
||||||
|
|
||||||
|
|
||||||
|
# Trim out all falsy values from an array.
|
||||||
|
_.compact = (array) -> item for item in array when item
|
||||||
|
|
||||||
|
|
||||||
|
# Return a completely flattened version of an array.
|
||||||
|
_.flatten = (array) ->
|
||||||
|
_.reduce array, (memo, value) ->
|
||||||
|
return memo.concat(_.flatten(value)) if _.isArray value
|
||||||
|
memo.push value
|
||||||
|
memo
|
||||||
|
, []
|
||||||
|
|
||||||
|
|
||||||
|
# Return a version of the array that does not contain the specified value(s).
|
||||||
|
_.without = (array) ->
|
||||||
|
values = _.rest arguments
|
||||||
|
val for val in _.toArray(array) when not _.include values, val
|
||||||
|
|
||||||
|
|
||||||
|
# Produce a duplicate-free version of the array. If the array has already
|
||||||
|
# been sorted, you have the option of using a faster algorithm.
|
||||||
|
_.uniq = (array, isSorted) ->
|
||||||
|
memo = []
|
||||||
|
for el, i in _.toArray array
|
||||||
|
memo.push el if i is 0 || (if isSorted is true then _.last(memo) isnt el else not _.include(memo, el))
|
||||||
|
memo
|
||||||
|
|
||||||
|
|
||||||
|
# Produce an array that contains every item shared between all the
|
||||||
|
# passed-in arrays.
|
||||||
|
_.intersect = (array) ->
|
||||||
|
rest = _.rest arguments
|
||||||
|
_.select _.uniq(array), (item) ->
|
||||||
|
_.all rest, (other) ->
|
||||||
|
_.indexOf(other, item) >= 0
|
||||||
|
|
||||||
|
|
||||||
|
# Zip together multiple lists into a single array -- elements that share
|
||||||
|
# an index go together.
|
||||||
|
_.zip = ->
|
||||||
|
length = _.max _.pluck arguments, 'length'
|
||||||
|
results = new Array length
|
||||||
|
for i in [0...length]
|
||||||
|
results[i] = _.pluck arguments, String i
|
||||||
|
results
|
||||||
|
|
||||||
|
|
||||||
|
# If the browser doesn't supply us with **indexOf** (I'm looking at you, MSIE),
|
||||||
|
# we need this function. Return the position of the first occurrence of an
|
||||||
|
# item in an array, or -1 if the item is not included in the array.
|
||||||
|
_.indexOf = (array, item) ->
|
||||||
|
return array.indexOf item if nativeIndexOf and array.indexOf is nativeIndexOf
|
||||||
|
i = 0; l = array.length
|
||||||
|
while l - i
|
||||||
|
if array[i] is item then return i else i++
|
||||||
|
-1
|
||||||
|
|
||||||
|
|
||||||
|
# Provide JavaScript 1.6's **lastIndexOf**, delegating to the native function,
|
||||||
|
# if possible.
|
||||||
|
_.lastIndexOf = (array, item) ->
|
||||||
|
return array.lastIndexOf(item) if nativeLastIndexOf and array.lastIndexOf is nativeLastIndexOf
|
||||||
|
i = array.length
|
||||||
|
while i
|
||||||
|
if array[i] is item then return i else i--
|
||||||
|
-1
|
||||||
|
|
||||||
|
|
||||||
|
# Generate an integer Array containing an arithmetic progression. A port of
|
||||||
|
# [the native Python **range** function](http://docs.python.org/library/functions.html#range).
|
||||||
|
_.range = (start, stop, step) ->
|
||||||
|
a = arguments
|
||||||
|
solo = a.length <= 1
|
||||||
|
i = start = if solo then 0 else a[0]
|
||||||
|
stop = if solo then a[0] else a[1]
|
||||||
|
step = a[2] or 1
|
||||||
|
len = Math.ceil((stop - start) / step)
|
||||||
|
return [] if len <= 0
|
||||||
|
range = new Array len
|
||||||
|
idx = 0
|
||||||
|
loop
|
||||||
|
return range if (if step > 0 then i - stop else stop - i) >= 0
|
||||||
|
range[idx] = i
|
||||||
|
idx++
|
||||||
|
i+= step
|
||||||
|
|
||||||
|
|
||||||
|
# Function Functions
|
||||||
|
# ------------------
|
||||||
|
|
||||||
|
# Create a function bound to a given object (assigning `this`, and arguments,
|
||||||
|
# optionally). Binding with arguments is also known as **curry**.
|
||||||
|
_.bind = (func, obj) ->
|
||||||
|
args = _.rest arguments, 2
|
||||||
|
-> func.apply obj or root, args.concat arguments
|
||||||
|
|
||||||
|
|
||||||
|
# Bind all of an object's methods to that object. Useful for ensuring that
|
||||||
|
# all callbacks defined on an object belong to it.
|
||||||
|
_.bindAll = (obj) ->
|
||||||
|
funcs = if arguments.length > 1 then _.rest(arguments) else _.functions(obj)
|
||||||
|
_.each funcs, (f) -> obj[f] = _.bind obj[f], obj
|
||||||
|
obj
|
||||||
|
|
||||||
|
|
||||||
|
# Delays a function for the given number of milliseconds, and then calls
|
||||||
|
# it with the arguments supplied.
|
||||||
|
_.delay = (func, wait) ->
|
||||||
|
args = _.rest arguments, 2
|
||||||
|
setTimeout((-> func.apply(func, args)), wait)
|
||||||
|
|
||||||
|
|
||||||
|
# Memoize an expensive function by storing its results.
|
||||||
|
_.memoize = (func, hasher) ->
|
||||||
|
memo = {}
|
||||||
|
hasher or= _.identity
|
||||||
|
->
|
||||||
|
key = hasher.apply this, arguments
|
||||||
|
return memo[key] if key of memo
|
||||||
|
memo[key] = func.apply this, arguments
|
||||||
|
|
||||||
|
|
||||||
|
# Defers a function, scheduling it to run after the current call stack has
|
||||||
|
# cleared.
|
||||||
|
_.defer = (func) ->
|
||||||
|
_.delay.apply _, [func, 1].concat _.rest arguments
|
||||||
|
|
||||||
|
|
||||||
|
# Returns the first function passed as an argument to the second,
|
||||||
|
# allowing you to adjust arguments, run code before and after, and
|
||||||
|
# conditionally execute the original function.
|
||||||
|
_.wrap = (func, wrapper) ->
|
||||||
|
-> wrapper.apply wrapper, [func].concat arguments
|
||||||
|
|
||||||
|
|
||||||
|
# Returns a function that is the composition of a list of functions, each
|
||||||
|
# consuming the return value of the function that follows.
|
||||||
|
_.compose = ->
|
||||||
|
funcs = arguments
|
||||||
|
->
|
||||||
|
args = arguments
|
||||||
|
for i in [funcs.length - 1..0] by -1
|
||||||
|
args = [funcs[i].apply(this, args)]
|
||||||
|
args[0]
|
||||||
|
|
||||||
|
|
||||||
|
# Object Functions
|
||||||
|
# ----------------
|
||||||
|
|
||||||
|
# Retrieve the names of an object's properties.
|
||||||
|
_.keys = nativeKeys or (obj) ->
|
||||||
|
return _.range 0, obj.length if _.isArray(obj)
|
||||||
|
key for key, val of obj
|
||||||
|
|
||||||
|
|
||||||
|
# Retrieve the values of an object's properties.
|
||||||
|
_.values = (obj) ->
|
||||||
|
_.map obj, _.identity
|
||||||
|
|
||||||
|
|
||||||
|
# Return a sorted list of the function names available in Underscore.
|
||||||
|
_.functions = (obj) ->
|
||||||
|
_.filter(_.keys(obj), (key) -> _.isFunction(obj[key])).sort()
|
||||||
|
|
||||||
|
|
||||||
|
# Extend a given object with all of the properties in a source object.
|
||||||
|
_.extend = (obj) ->
|
||||||
|
for source in _.rest(arguments)
|
||||||
|
obj[key] = val for key, val of source
|
||||||
|
obj
|
||||||
|
|
||||||
|
|
||||||
|
# Create a (shallow-cloned) duplicate of an object.
|
||||||
|
_.clone = (obj) ->
|
||||||
|
return obj.slice 0 if _.isArray obj
|
||||||
|
_.extend {}, obj
|
||||||
|
|
||||||
|
|
||||||
|
# Invokes interceptor with the obj, and then returns obj.
|
||||||
|
# The primary purpose of this method is to "tap into" a method chain,
|
||||||
|
# in order to perform operations on intermediate results within
|
||||||
|
the chain.
|
||||||
|
_.tap = (obj, interceptor) ->
|
||||||
|
interceptor obj
|
||||||
|
obj
|
||||||
|
|
||||||
|
|
||||||
|
# Perform a deep comparison to check if two objects are equal.
|
||||||
|
_.isEqual = (a, b) ->
|
||||||
|
# Check object identity.
|
||||||
|
return true if a is b
|
||||||
|
# Different types?
|
||||||
|
atype = typeof(a); btype = typeof(b)
|
||||||
|
return false if atype isnt btype
|
||||||
|
# Basic equality test (watch out for coercions).
|
||||||
|
return true if `a == b`
|
||||||
|
# One is falsy and the other truthy.
|
||||||
|
return false if (!a and b) or (a and !b)
|
||||||
|
# One of them implements an `isEqual()`?
|
||||||
|
return a.isEqual(b) if a.isEqual
|
||||||
|
# Check dates' integer values.
|
||||||
|
return a.getTime() is b.getTime() if _.isDate(a) and _.isDate(b)
|
||||||
|
# Both are NaN?
|
||||||
|
return false if _.isNaN(a) and _.isNaN(b)
|
||||||
|
# Compare regular expressions.
|
||||||
|
if _.isRegExp(a) and _.isRegExp(b)
|
||||||
|
return a.source is b.source and
|
||||||
|
a.global is b.global and
|
||||||
|
a.ignoreCase is b.ignoreCase and
|
||||||
|
a.multiline is b.multiline
|
||||||
|
# If a is not an object by this point, we can't handle it.
|
||||||
|
return false if atype isnt 'object'
|
||||||
|
# Check for different array lengths before comparing contents.
|
||||||
|
return false if a.length and (a.length isnt b.length)
|
||||||
|
# Nothing else worked, deep compare the contents.
|
||||||
|
aKeys = _.keys(a); bKeys = _.keys(b)
|
||||||
|
# Different object sizes?
|
||||||
|
return false if aKeys.length isnt bKeys.length
|
||||||
|
# Recursive comparison of contents.
|
||||||
|
return false for key, val of a when !(key of b) or !_.isEqual(val, b[key])
|
||||||
|
true
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given array or object empty?
|
||||||
|
_.isEmpty = (obj) ->
|
||||||
|
return obj.length is 0 if _.isArray(obj) or _.isString(obj)
|
||||||
|
return false for own key of obj
|
||||||
|
true
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value a DOM element?
|
||||||
|
_.isElement = (obj) -> obj and obj.nodeType is 1
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value an array?
|
||||||
|
_.isArray = nativeIsArray or (obj) -> !!(obj and obj.concat and obj.unshift and not obj.callee)
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given variable an arguments object?
|
||||||
|
_.isArguments = (obj) -> obj and obj.callee
|
||||||
|
|
||||||
|
|
||||||
|
# Is the given value a function?
|
||||||
|
_.isFunction = (obj) -> !!(obj and obj.constructor and obj.call and obj.apply)
|
||||||
|
|
||||||
|
|
||||||
|
# Is the given value a string?
|
||||||
|
_.isString = (obj) -> !!(obj is '' or (obj and obj.charCodeAt and obj.substr))
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value a number?
|
||||||
|
_.isNumber = (obj) -> (obj is +obj) or toString.call(obj) is '[object Number]'
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value a boolean?
|
||||||
|
_.isBoolean = (obj) -> obj is true or obj is false
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value a Date?
|
||||||
|
_.isDate = (obj) -> !!(obj and obj.getTimezoneOffset and obj.setUTCFullYear)
|
||||||
|
|
||||||
|
|
||||||
|
# Is the given value a regular expression?
|
||||||
|
_.isRegExp = (obj) -> !!(obj and obj.exec and (obj.ignoreCase or obj.ignoreCase is false))
|
||||||
|
|
||||||
|
|
||||||
|
# Is the given value NaN -- this one is interesting. `NaN != NaN`, and
|
||||||
|
# `isNaN(undefined) == true`, so we make sure it's a number first.
|
||||||
|
_.isNaN = (obj) -> _.isNumber(obj) and window.isNaN(obj)
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given value equal to null?
|
||||||
|
_.isNull = (obj) -> obj is null
|
||||||
|
|
||||||
|
|
||||||
|
# Is a given variable undefined?
|
||||||
|
_.isUndefined = (obj) -> typeof obj is 'undefined'
|
||||||
|
|
||||||
|
|
||||||
|
# Utility Functions
|
||||||
|
# -----------------
|
||||||
|
|
||||||
|
# Run Underscore.js in noConflict mode, returning the `_` variable to its
|
||||||
|
# previous owner. Returns a reference to the Underscore object.
|
||||||
|
_.noConflict = ->
|
||||||
|
root._ = previousUnderscore
|
||||||
|
this
|
||||||
|
|
||||||
|
|
||||||
|
# Keep the identity function around for default iterators.
|
||||||
|
_.identity = (value) -> value
|
||||||
|
|
||||||
|
|
||||||
|
# Run a function `n` times.
|
||||||
|
_.times = (n, iterator, context) ->
|
||||||
|
iterator.call context, i for i in [0...n]
|
||||||
|
|
||||||
|
|
||||||
|
# Break out of the middle of an iteration.
|
||||||
|
_.breakLoop = -> throw breaker
|
||||||
|
|
||||||
|
|
||||||
|
# Add your own custom functions to the Underscore object, ensuring that
|
||||||
|
# they're correctly added to the OOP wrapper as well.
|
||||||
|
_.mixin = (obj) ->
|
||||||
|
for name in _.functions(obj)
|
||||||
|
addToWrapper name, _[name] = obj[name]
|
||||||
|
|
||||||
|
|
||||||
|
# Generate a unique integer id (unique within the entire client session).
|
||||||
|
# Useful for temporary DOM ids.
|
||||||
|
idCounter = 0
|
||||||
|
_.uniqueId = (prefix) ->
|
||||||
|
(prefix or '') + idCounter++
|
||||||
|
|
||||||
|
|
||||||
|
# By default, Underscore uses **ERB**-style template delimiters, change the
|
||||||
|
# following template settings to use alternative delimiters.
|
||||||
|
_.templateSettings = {
|
||||||
|
start: '<%'
|
||||||
|
end: '%>'
|
||||||
|
interpolate: /<%=(.+?)%>/g
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# JavaScript templating a-la **ERB**, pilfered from John Resig's
|
||||||
|
# *Secrets of the JavaScript Ninja*, page 83.
|
||||||
|
# Single-quote fix from Rick Strahl.
|
||||||
|
# With alterations for arbitrary delimiters, and to preserve whitespace.
|
||||||
|
_.template = (str, data) ->
|
||||||
|
c = _.templateSettings
|
||||||
|
endMatch = new RegExp("'(?=[^"+c.end.substr(0, 1)+"]*"+escapeRegExp(c.end)+")","g")
|
||||||
|
fn = new Function 'obj',
|
||||||
|
'var p=[],print=function(){p.push.apply(p,arguments);};' +
|
||||||
|
'with(obj||{}){p.push(\'' +
|
||||||
|
str.replace(/\r/g, '\\r')
|
||||||
|
.replace(/\n/g, '\\n')
|
||||||
|
.replace(/\t/g, '\\t')
|
||||||
|
.replace(endMatch,"<22><><EFBFBD>")
|
||||||
|
.split("'").join("\\'")
|
||||||
|
.split("<22><><EFBFBD>").join("'")
|
||||||
|
.replace(c.interpolate, "',$1,'")
|
||||||
|
.split(c.start).join("');")
|
||||||
|
.split(c.end).join("p.push('") +
|
||||||
|
"');}return p.join('');"
|
||||||
|
if data then fn(data) else fn
|
||||||
|
|
||||||
|
|
||||||
|
# Aliases
|
||||||
|
# -------
|
||||||
|
|
||||||
|
_.forEach = _.each
|
||||||
|
_.foldl = _.inject = _.reduce
|
||||||
|
_.foldr = _.reduceRight
|
||||||
|
_.select = _.filter
|
||||||
|
_.all = _.every
|
||||||
|
_.any = _.some
|
||||||
|
_.contains = _.include
|
||||||
|
_.head = _.first
|
||||||
|
_.tail = _.rest
|
||||||
|
_.methods = _.functions
|
||||||
|
|
||||||
|
|
||||||
|
# Setup the OOP Wrapper
|
||||||
|
# ---------------------
|
||||||
|
|
||||||
|
# If Underscore is called as a function, it returns a wrapped object that
|
||||||
|
# can be used OO-style. This wrapper holds altered versions of all the
|
||||||
|
# underscore functions. Wrapped objects may be chained.
|
||||||
|
wrapper = (obj) ->
|
||||||
|
this._wrapped = obj
|
||||||
|
this
|
||||||
|
|
||||||
|
|
||||||
|
# Helper function to continue chaining intermediate results.
|
||||||
|
result = (obj, chain) ->
|
||||||
|
if chain then _(obj).chain() else obj
|
||||||
|
|
||||||
|
|
||||||
|
# A method to easily add functions to the OOP wrapper.
|
||||||
|
addToWrapper = (name, func) ->
|
||||||
|
wrapper.prototype[name] = ->
|
||||||
|
args = _.toArray arguments
|
||||||
|
unshift.call args, this._wrapped
|
||||||
|
result func.apply(_, args), this._chain
|
||||||
|
|
||||||
|
|
||||||
|
# Add all ofthe Underscore functions to the wrapper object.
|
||||||
|
_.mixin _
|
||||||
|
|
||||||
|
|
||||||
|
# Add all mutator Array functions to the wrapper.
|
||||||
|
_.each ['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], (name) ->
|
||||||
|
method = Array.prototype[name]
|
||||||
|
wrapper.prototype[name] = ->
|
||||||
|
method.apply(this._wrapped, arguments)
|
||||||
|
result(this._wrapped, this._chain)
|
||||||
|
|
||||||
|
|
||||||
|
# Add all accessor Array functions to the wrapper.
|
||||||
|
_.each ['concat', 'join', 'slice'], (name) ->
|
||||||
|
method = Array.prototype[name]
|
||||||
|
wrapper.prototype[name] = ->
|
||||||
|
result(method.apply(this._wrapped, arguments), this._chain)
|
||||||
|
|
||||||
|
|
||||||
|
# Start chaining a wrapped Underscore object.
|
||||||
|
wrapper::chain = ->
|
||||||
|
this._chain = true
|
||||||
|
this
|
||||||
|
|
||||||
|
|
||||||
|
# Extracts the result from a wrapped and chained object.
|
||||||
|
wrapper::value = -> this._wrapped
|
||||||
|
</textarea></form>
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/x-coffeescript</code>.</p>
|
||||||
|
|
||||||
|
<p>The CoffeeScript mode was written by Jeff Pickhardt (<a href="LICENSE">license</a>).</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
101
codemirror-2.36/mode/commonlisp/commonlisp.js
Normal file
101
codemirror-2.36/mode/commonlisp/commonlisp.js
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
CodeMirror.defineMode("commonlisp", function (config) {
|
||||||
|
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") return "atom";
|
||||||
|
if (name.charAt(0) == ":") 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}, 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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-common-lisp", "commonlisp");
|
165
codemirror-2.36/mode/commonlisp/index.html
Normal file
165
codemirror-2.36/mode/commonlisp/index.html
Normal file
|
@ -0,0 +1,165 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Common Lisp mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="commonlisp.js"></script>
|
||||||
|
<style>.CodeMirror {background: #f8f8f8;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Common Lisp mode</h1>
|
||||||
|
<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>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
448
codemirror-2.36/mode/css/css.js
Normal file
448
codemirror-2.36/mode/css/css.js
Normal file
|
@ -0,0 +1,448 @@
|
||||||
|
CodeMirror.defineMode("css", function(config) {
|
||||||
|
var indentUnit = config.indentUnit, type;
|
||||||
|
|
||||||
|
var atMediaTypes = keySet([
|
||||||
|
"all", "aural", "braille", "handheld", "print", "projection", "screen",
|
||||||
|
"tty", "tv", "embossed"
|
||||||
|
]);
|
||||||
|
|
||||||
|
var atMediaFeatures = keySet([
|
||||||
|
"width", "min-width", "max-width", "height", "min-height", "max-height",
|
||||||
|
"device-width", "min-device-width", "max-device-width", "device-height",
|
||||||
|
"min-device-height", "max-device-height", "aspect-ratio",
|
||||||
|
"min-aspect-ratio", "max-aspect-ratio", "device-aspect-ratio",
|
||||||
|
"min-device-aspect-ratio", "max-device-aspect-ratio", "color", "min-color",
|
||||||
|
"max-color", "color-index", "min-color-index", "max-color-index",
|
||||||
|
"monochrome", "min-monochrome", "max-monochrome", "resolution",
|
||||||
|
"min-resolution", "max-resolution", "scan", "grid"
|
||||||
|
]);
|
||||||
|
|
||||||
|
var propertyKeywords = keySet([
|
||||||
|
"align-content", "align-items", "align-self", "alignment-adjust",
|
||||||
|
"alignment-baseline", "anchor-point", "animation", "animation-delay",
|
||||||
|
"animation-direction", "animation-duration", "animation-iteration-count",
|
||||||
|
"animation-name", "animation-play-state", "animation-timing-function",
|
||||||
|
"appearance", "azimuth", "backface-visibility", "background",
|
||||||
|
"background-attachment", "background-clip", "background-color",
|
||||||
|
"background-image", "background-origin", "background-position",
|
||||||
|
"background-repeat", "background-size", "baseline-shift", "binding",
|
||||||
|
"bleed", "bookmark-label", "bookmark-level", "bookmark-state",
|
||||||
|
"bookmark-target", "border", "border-bottom", "border-bottom-color",
|
||||||
|
"border-bottom-left-radius", "border-bottom-right-radius",
|
||||||
|
"border-bottom-style", "border-bottom-width", "border-collapse",
|
||||||
|
"border-color", "border-image", "border-image-outset",
|
||||||
|
"border-image-repeat", "border-image-slice", "border-image-source",
|
||||||
|
"border-image-width", "border-left", "border-left-color",
|
||||||
|
"border-left-style", "border-left-width", "border-radius", "border-right",
|
||||||
|
"border-right-color", "border-right-style", "border-right-width",
|
||||||
|
"border-spacing", "border-style", "border-top", "border-top-color",
|
||||||
|
"border-top-left-radius", "border-top-right-radius", "border-top-style",
|
||||||
|
"border-top-width", "border-width", "bottom", "box-decoration-break",
|
||||||
|
"box-shadow", "box-sizing", "break-after", "break-before", "break-inside",
|
||||||
|
"caption-side", "clear", "clip", "color", "color-profile", "column-count",
|
||||||
|
"column-fill", "column-gap", "column-rule", "column-rule-color",
|
||||||
|
"column-rule-style", "column-rule-width", "column-span", "column-width",
|
||||||
|
"columns", "content", "counter-increment", "counter-reset", "crop", "cue",
|
||||||
|
"cue-after", "cue-before", "cursor", "direction", "display",
|
||||||
|
"dominant-baseline", "drop-initial-after-adjust",
|
||||||
|
"drop-initial-after-align", "drop-initial-before-adjust",
|
||||||
|
"drop-initial-before-align", "drop-initial-size", "drop-initial-value",
|
||||||
|
"elevation", "empty-cells", "fit", "fit-position", "flex", "flex-basis",
|
||||||
|
"flex-direction", "flex-flow", "flex-grow", "flex-shrink", "flex-wrap",
|
||||||
|
"float", "float-offset", "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-cell", "grid-column", "grid-column-align",
|
||||||
|
"grid-column-sizing", "grid-column-span", "grid-columns", "grid-flow",
|
||||||
|
"grid-row", "grid-row-align", "grid-row-sizing", "grid-row-span",
|
||||||
|
"grid-rows", "grid-template", "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", "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", "rendering-intent",
|
||||||
|
"resize", "rest", "rest-after", "rest-before", "richness", "right",
|
||||||
|
"rotation", "rotation-point", "ruby-align", "ruby-overhang",
|
||||||
|
"ruby-position", "ruby-span", "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-shadow",
|
||||||
|
"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"
|
||||||
|
]);
|
||||||
|
|
||||||
|
var colorKeywords = keySet([
|
||||||
|
"black", "silver", "gray", "white", "maroon", "red", "purple", "fuchsia",
|
||||||
|
"green", "lime", "olive", "yellow", "navy", "blue", "teal", "aqua"
|
||||||
|
]);
|
||||||
|
|
||||||
|
var valueKeywords = keySet([
|
||||||
|
"above", "absolute", "activeborder", "activecaption", "afar",
|
||||||
|
"after-white-space", "ahead", "alias", "all", "all-scroll", "alternate",
|
||||||
|
"always", "amharic", "amharic-abegede", "antialiased", "appworkspace",
|
||||||
|
"arabic-indic", "armenian", "asterisks", "auto", "avoid", "background",
|
||||||
|
"backwards", "baseline", "below", "bidi-override", "binary", "bengali",
|
||||||
|
"blink", "block", "block-axis", "bold", "bolder", "border", "border-box",
|
||||||
|
"both", "bottom", "break-all", "break-word", "button", "button-bevel",
|
||||||
|
"buttonface", "buttonhighlight", "buttonshadow", "buttontext", "cambodian",
|
||||||
|
"capitalize", "caps-lock-indicator", "caption", "captiontext", "caret",
|
||||||
|
"cell", "center", "checkbox", "circle", "cjk-earthly-branch",
|
||||||
|
"cjk-heavenly-stem", "cjk-ideographic", "clear", "clip", "close-quote",
|
||||||
|
"col-resize", "collapse", "compact", "condensed", "contain", "content",
|
||||||
|
"content-box", "context-menu", "continuous", "copy", "cover", "crop",
|
||||||
|
"cross", "crosshair", "currentcolor", "cursive", "dashed", "decimal",
|
||||||
|
"decimal-leading-zero", "default", "default-button", "destination-atop",
|
||||||
|
"destination-in", "destination-out", "destination-over", "devanagari",
|
||||||
|
"disc", "discard", "document", "dot-dash", "dot-dot-dash", "dotted",
|
||||||
|
"double", "down", "e-resize", "ease", "ease-in", "ease-in-out", "ease-out",
|
||||||
|
"element", "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", "ew-resize", "expanded", "extra-condensed",
|
||||||
|
"extra-expanded", "fantasy", "fast", "fill", "fixed", "flat", "footnotes",
|
||||||
|
"forwards", "from", "geometricPrecision", "georgian", "graytext", "groove",
|
||||||
|
"gujarati", "gurmukhi", "hand", "hangul", "hangul-consonant", "hebrew",
|
||||||
|
"help", "hidden", "hide", "higher", "highlight", "highlighttext",
|
||||||
|
"hiragana", "hiragana-iroha", "horizontal", "hsl", "hsla", "icon", "ignore",
|
||||||
|
"inactiveborder", "inactivecaption", "inactivecaptiontext", "infinite",
|
||||||
|
"infobackground", "infotext", "inherit", "initial", "inline", "inline-axis",
|
||||||
|
"inline-block", "inline-table", "inset", "inside", "intrinsic", "invert",
|
||||||
|
"italic", "justify", "kannada", "katakana", "katakana-iroha", "khmer",
|
||||||
|
"landscape", "lao", "large", "larger", "left", "level", "lighter",
|
||||||
|
"line-through", "linear", "lines", "list-item", "listbox", "listitem",
|
||||||
|
"local", "logical", "loud", "lower", "lower-alpha", "lower-armenian",
|
||||||
|
"lower-greek", "lower-hexadecimal", "lower-latin", "lower-norwegian",
|
||||||
|
"lower-roman", "lowercase", "ltr", "malayalam", "match",
|
||||||
|
"media-controls-background", "media-current-time-display",
|
||||||
|
"media-fullscreen-button", "media-mute-button", "media-play-button",
|
||||||
|
"media-return-to-realtime-button", "media-rewind-button",
|
||||||
|
"media-seek-back-button", "media-seek-forward-button", "media-slider",
|
||||||
|
"media-sliderthumb", "media-time-remaining-display", "media-volume-slider",
|
||||||
|
"media-volume-slider-container", "media-volume-sliderthumb", "medium",
|
||||||
|
"menu", "menulist", "menulist-button", "menulist-text",
|
||||||
|
"menulist-textfield", "menutext", "message-box", "middle", "min-intrinsic",
|
||||||
|
"mix", "mongolian", "monospace", "move", "multiple", "myanmar", "n-resize",
|
||||||
|
"narrower", "navy", "ne-resize", "nesw-resize", "no-close-quote", "no-drop",
|
||||||
|
"no-open-quote", "no-repeat", "none", "normal", "not-allowed", "nowrap",
|
||||||
|
"ns-resize", "nw-resize", "nwse-resize", "oblique", "octal", "open-quote",
|
||||||
|
"optimizeLegibility", "optimizeSpeed", "oriya", "oromo", "outset",
|
||||||
|
"outside", "overlay", "overline", "padding", "padding-box", "painted",
|
||||||
|
"paused", "persian", "plus-darker", "plus-lighter", "pointer", "portrait",
|
||||||
|
"pre", "pre-line", "pre-wrap", "preserve-3d", "progress", "push-button",
|
||||||
|
"radio", "read-only", "read-write", "read-write-plaintext-only", "relative",
|
||||||
|
"repeat", "repeat-x", "repeat-y", "reset", "reverse", "rgb", "rgba",
|
||||||
|
"ridge", "right", "round", "row-resize", "rtl", "run-in", "running",
|
||||||
|
"s-resize", "sans-serif", "scroll", "scrollbar", "se-resize", "searchfield",
|
||||||
|
"searchfield-cancel-button", "searchfield-decoration",
|
||||||
|
"searchfield-results-button", "searchfield-results-decoration",
|
||||||
|
"semi-condensed", "semi-expanded", "separate", "serif", "show", "sidama",
|
||||||
|
"single", "skip-white-space", "slide", "slider-horizontal",
|
||||||
|
"slider-vertical", "sliderthumb-horizontal", "sliderthumb-vertical", "slow",
|
||||||
|
"small", "small-caps", "small-caption", "smaller", "solid", "somali",
|
||||||
|
"source-atop", "source-in", "source-out", "source-over", "space", "square",
|
||||||
|
"square-button", "start", "static", "status-bar", "stretch", "stroke",
|
||||||
|
"sub", "subpixel-antialiased", "super", "sw-resize", "table",
|
||||||
|
"table-caption", "table-cell", "table-column", "table-column-group",
|
||||||
|
"table-footer-group", "table-header-group", "table-row", "table-row-group",
|
||||||
|
"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",
|
||||||
|
"transparent", "ultra-condensed", "ultra-expanded", "underline", "up",
|
||||||
|
"upper-alpha", "upper-armenian", "upper-greek", "upper-hexadecimal",
|
||||||
|
"upper-latin", "upper-norwegian", "upper-roman", "uppercase", "urdu", "url",
|
||||||
|
"vertical", "vertical-text", "visible", "visibleFill", "visiblePainted",
|
||||||
|
"visibleStroke", "visual", "w-resize", "wait", "wave", "white", "wider",
|
||||||
|
"window", "windowframe", "windowtext", "x-large", "x-small", "xor",
|
||||||
|
"xx-large", "xx-small", "yellow"
|
||||||
|
]);
|
||||||
|
|
||||||
|
function keySet(array) { var keys = {}; for (var i = 0; i < array.length; ++i) keys[array[i]] = true; return keys; }
|
||||||
|
function ret(style, tp) {type = tp; return style;}
|
||||||
|
|
||||||
|
function tokenBase(stream, state) {
|
||||||
|
var ch = stream.next();
|
||||||
|
if (ch == "@") {stream.eatWhile(/[\w\\\-]/); return ret("def", stream.current());}
|
||||||
|
else if (ch == "/" && stream.eat("*")) {
|
||||||
|
state.tokenize = tokenCComment;
|
||||||
|
return tokenCComment(stream, state);
|
||||||
|
}
|
||||||
|
else if (ch == "<" && stream.eat("!")) {
|
||||||
|
state.tokenize = tokenSGMLComment;
|
||||||
|
return tokenSGMLComment(stream, state);
|
||||||
|
}
|
||||||
|
else if (ch == "=") ret(null, "compare");
|
||||||
|
else if ((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)) {
|
||||||
|
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(/^[^-]+-/)) {
|
||||||
|
return ret("meta", type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (/[,+>*\/]/.test(ch)) {
|
||||||
|
return ret(null, "select-op");
|
||||||
|
}
|
||||||
|
else if (ch == "." && stream.match(/^-?[_a-z][_a-z0-9-]*/i)) {
|
||||||
|
return ret("qualifier", type);
|
||||||
|
}
|
||||||
|
else if (ch == ":") {
|
||||||
|
return ret("operator", ch);
|
||||||
|
}
|
||||||
|
else if (/[;{}\[\]\(\)]/.test(ch)) {
|
||||||
|
return ret(null, ch);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
stream.eatWhile(/[\w\\\-]/);
|
||||||
|
return ret("property", "variable");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function tokenCComment(stream, state) {
|
||||||
|
var maybeEnd = false, ch;
|
||||||
|
while ((ch = stream.next()) != null) {
|
||||||
|
if (maybeEnd && ch == "/") {
|
||||||
|
state.tokenize = tokenBase;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
maybeEnd = (ch == "*");
|
||||||
|
}
|
||||||
|
return ret("comment", "comment");
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
|
break;
|
||||||
|
escaped = !escaped && ch == "\\";
|
||||||
|
}
|
||||||
|
if (!escaped) state.tokenize = tokenBase;
|
||||||
|
return ret("string", "string");
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState: function(base) {
|
||||||
|
return {tokenize: tokenBase,
|
||||||
|
baseIndent: base || 0,
|
||||||
|
stack: []};
|
||||||
|
},
|
||||||
|
|
||||||
|
token: function(stream, state) {
|
||||||
|
|
||||||
|
// Use these terms when applicable (see http://www.xanthir.com/blog/b4E50)
|
||||||
|
//
|
||||||
|
// rule** or **ruleset:
|
||||||
|
// A selector + braces combo, or an at-rule.
|
||||||
|
//
|
||||||
|
// declaration block:
|
||||||
|
// A sequence of declarations.
|
||||||
|
//
|
||||||
|
// declaration:
|
||||||
|
// A property + colon + value combo.
|
||||||
|
//
|
||||||
|
// property value:
|
||||||
|
// The entire value of a property.
|
||||||
|
//
|
||||||
|
// component value:
|
||||||
|
// A single piece of a property value. Like the 5px in
|
||||||
|
// text-shadow: 0 0 5px blue;. Can also refer to things that are
|
||||||
|
// multiple terms, like the 1-4 terms that make up the background-size
|
||||||
|
// portion of the background shorthand.
|
||||||
|
//
|
||||||
|
// term:
|
||||||
|
// The basic unit of author-facing CSS, like a single number (5),
|
||||||
|
// dimension (5px), string ("foo"), or function. Officially defined
|
||||||
|
// by the CSS 2.1 grammar (look for the 'term' production)
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// simple selector:
|
||||||
|
// A single atomic selector, like a type selector, an attr selector, a
|
||||||
|
// class selector, etc.
|
||||||
|
//
|
||||||
|
// compound selector:
|
||||||
|
// One or more simple selectors without a combinator. div.example is
|
||||||
|
// compound, div > .example is not.
|
||||||
|
//
|
||||||
|
// complex selector:
|
||||||
|
// One or more compound selectors chained with combinators.
|
||||||
|
//
|
||||||
|
// combinator:
|
||||||
|
// The parts of selectors that express relationships. There are four
|
||||||
|
// currently - the space (descendant combinator), the greater-than
|
||||||
|
// bracket (child combinator), the plus sign (next sibling combinator),
|
||||||
|
// and the tilda (following sibling combinator).
|
||||||
|
//
|
||||||
|
// sequence of selectors:
|
||||||
|
// One or more of the named type of selector chained with commas.
|
||||||
|
|
||||||
|
if (stream.eatSpace()) return null;
|
||||||
|
var style = state.tokenize(stream, state);
|
||||||
|
|
||||||
|
// Changing style returned based on context
|
||||||
|
var context = state.stack[state.stack.length-1];
|
||||||
|
if (style == "property") {
|
||||||
|
if (context == "propertyValue"){
|
||||||
|
if (valueKeywords[stream.current()]) {
|
||||||
|
style = "string-2";
|
||||||
|
} else if (colorKeywords[stream.current()]) {
|
||||||
|
style = "keyword";
|
||||||
|
} else {
|
||||||
|
style = "variable-2";
|
||||||
|
}
|
||||||
|
} else if (context == "rule") {
|
||||||
|
if (!propertyKeywords[stream.current()]) {
|
||||||
|
style += " error";
|
||||||
|
}
|
||||||
|
} else if (!context || context == "@media{") {
|
||||||
|
style = "tag";
|
||||||
|
} else if (context == "@media") {
|
||||||
|
if (atMediaTypes[stream.current()]) {
|
||||||
|
style = "attribute"; // Known attribute
|
||||||
|
} else if (/^(only|not)$/i.test(stream.current())) {
|
||||||
|
style = "keyword";
|
||||||
|
} else if (stream.current().toLowerCase() == "and") {
|
||||||
|
style = "error"; // "and" is only allowed in @mediaType
|
||||||
|
} else if (atMediaFeatures[stream.current()]) {
|
||||||
|
style = "error"; // Known property, should be in @mediaType(
|
||||||
|
} else {
|
||||||
|
// Unknown, expecting keyword or attribute, assuming attribute
|
||||||
|
style = "attribute error";
|
||||||
|
}
|
||||||
|
} else if (context == "@mediaType") {
|
||||||
|
if (atMediaTypes[stream.current()]) {
|
||||||
|
style = "attribute";
|
||||||
|
} else if (stream.current().toLowerCase() == "and") {
|
||||||
|
style = "operator";
|
||||||
|
} else if (/^(only|not)$/i.test(stream.current())) {
|
||||||
|
style = "error"; // Only allowed in @media
|
||||||
|
} else if (atMediaFeatures[stream.current()]) {
|
||||||
|
style = "error"; // Known property, should be in parentheses
|
||||||
|
} else {
|
||||||
|
// Unknown attribute or property, but expecting property (preceded
|
||||||
|
// by "and"). Should be in parentheses
|
||||||
|
style = "error";
|
||||||
|
}
|
||||||
|
} else if (context == "@mediaType(") {
|
||||||
|
if (propertyKeywords[stream.current()]) {
|
||||||
|
// do nothing, remains "property"
|
||||||
|
} else if (atMediaTypes[stream.current()]) {
|
||||||
|
style = "error"; // Known property, should be in parentheses
|
||||||
|
} else if (stream.current().toLowerCase() == "and") {
|
||||||
|
style = "operator";
|
||||||
|
} else if (/^(only|not)$/i.test(stream.current())) {
|
||||||
|
style = "error"; // Only allowed in @media
|
||||||
|
} else {
|
||||||
|
style += " error";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
style = "error";
|
||||||
|
}
|
||||||
|
} else if (style == "atom") {
|
||||||
|
if(!context || context == "@media{") {
|
||||||
|
style = "builtin";
|
||||||
|
} else if (context == "propertyValue") {
|
||||||
|
if (!/^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/.test(stream.current())) {
|
||||||
|
style += " error";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
style = "error";
|
||||||
|
}
|
||||||
|
} else if (context == "@media" && type == "{") {
|
||||||
|
style = "error";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Push/pop context stack
|
||||||
|
if (type == "{") {
|
||||||
|
if (context == "@media" || context == "@mediaType") {
|
||||||
|
state.stack.pop();
|
||||||
|
state.stack[state.stack.length-1] = "@media{";
|
||||||
|
}
|
||||||
|
else state.stack.push("rule");
|
||||||
|
}
|
||||||
|
else if (type == "}") {
|
||||||
|
state.stack.pop();
|
||||||
|
if (context == "propertyValue") state.stack.pop();
|
||||||
|
}
|
||||||
|
else if (type == "@media") state.stack.push("@media");
|
||||||
|
else if (context == "@media" && /\b(keyword|attribute)\b/.test(style))
|
||||||
|
state.stack.push("@mediaType");
|
||||||
|
else if (context == "@mediaType" && stream.current() == ",") state.stack.pop();
|
||||||
|
else if (context == "@mediaType" && type == "(") state.stack.push("@mediaType(");
|
||||||
|
else if (context == "@mediaType(" && type == ")") state.stack.pop();
|
||||||
|
else if (context == "rule" && type == ":") state.stack.push("propertyValue");
|
||||||
|
else if (context == "propertyValue" && type == ";") state.stack.pop();
|
||||||
|
return style;
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
var n = state.stack.length;
|
||||||
|
if (/^\}/.test(textAfter))
|
||||||
|
n -= state.stack[state.stack.length-1] == "propertyValue" ? 2 : 1;
|
||||||
|
return state.baseIndent + n * indentUnit;
|
||||||
|
},
|
||||||
|
|
||||||
|
electricChars: "}"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/css", "css");
|
58
codemirror-2.36/mode/css/index.html
Normal file
58
codemirror-2.36/mode/css/index.html
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: CSS mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="css.js"></script>
|
||||||
|
<style>.CodeMirror {background: #f8f8f8;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: CSS mode</h1>
|
||||||
|
<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"), {});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/css</code>.</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>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
501
codemirror-2.36/mode/css/test.js
Normal file
501
codemirror-2.36/mode/css/test.js
Normal file
|
@ -0,0 +1,501 @@
|
||||||
|
// Initiate ModeTest and set defaults
|
||||||
|
var MT = ModeTest;
|
||||||
|
MT.modeName = 'css';
|
||||||
|
MT.modeOptions = {};
|
||||||
|
|
||||||
|
// Requires at least one media query
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaEmpty',
|
||||||
|
'@media { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'error', '{',
|
||||||
|
null, ' }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaMultiple',
|
||||||
|
'@media not screen and (color), not print and (color) { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'keyword', 'not',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' (',
|
||||||
|
'property', 'color',
|
||||||
|
null, '), ',
|
||||||
|
'keyword', 'not',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'print',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' (',
|
||||||
|
'property', 'color',
|
||||||
|
null, ') { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaCheckStack',
|
||||||
|
'@media screen { } foo { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' { } ',
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaCheckStack',
|
||||||
|
'@media screen (color) { } foo { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' (',
|
||||||
|
'property', 'color',
|
||||||
|
null, ') { } ',
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaCheckStackInvalidAttribute',
|
||||||
|
'@media foobarhello { } foo { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute error', 'foobarhello',
|
||||||
|
null, ' { } ',
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "and" is only allowed immediately preceding a media expression
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidAttribute',
|
||||||
|
'@media foobarhello { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute error', 'foobarhello',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "and" is only allowed immediately preceding a media expression
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidAnd',
|
||||||
|
'@media and screen { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'error', 'and',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "not" is only allowed as the first item in each media query
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidNot',
|
||||||
|
'@media screen not (not) { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'error', 'not',
|
||||||
|
null, ' (',
|
||||||
|
'error', 'not',
|
||||||
|
null, ') { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "only" is only allowed as the first item in each media query
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidOnly',
|
||||||
|
'@media screen only (only) { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'error', 'only',
|
||||||
|
null, ' (',
|
||||||
|
'error', 'only',
|
||||||
|
null, ') { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "foobarhello" is neither a known type or property, but
|
||||||
|
// property was expected (after "and"), and it should be in parenthese.
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaUnknownType',
|
||||||
|
'@media screen and foobarhello { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' ',
|
||||||
|
'error', 'foobarhello',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "color" is not a known type, but is a known property, and
|
||||||
|
// should be in parentheses.
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidType',
|
||||||
|
'@media screen and color { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' ',
|
||||||
|
'error', 'color',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Error, because "print" is not a known property, but is a known type,
|
||||||
|
// and should not be in parenthese.
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaInvalidProperty',
|
||||||
|
'@media screen and (print) { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' (',
|
||||||
|
'error', 'print',
|
||||||
|
null, ') { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Soft error, because "foobarhello" is not a known property or type.
|
||||||
|
MT.testMode(
|
||||||
|
'atMediaUnknownProperty',
|
||||||
|
'@media screen and (foobarhello) { }',
|
||||||
|
[
|
||||||
|
'def', '@media',
|
||||||
|
null, ' ',
|
||||||
|
'attribute', 'screen',
|
||||||
|
null, ' ',
|
||||||
|
'operator', 'and',
|
||||||
|
null, ' (',
|
||||||
|
'property error', 'foobarhello',
|
||||||
|
null, ') { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagSelector',
|
||||||
|
'foo { }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'classSelector',
|
||||||
|
'.foo-bar_hello { }',
|
||||||
|
[
|
||||||
|
'qualifier', '.foo-bar_hello',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'idSelector',
|
||||||
|
'#foo { #foo }',
|
||||||
|
[
|
||||||
|
'builtin', '#foo',
|
||||||
|
null, ' { ',
|
||||||
|
'error', '#foo',
|
||||||
|
null, ' }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagSelectorUnclosed',
|
||||||
|
'foo { margin: 0 } bar { }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'margin',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '0',
|
||||||
|
null, ' } ',
|
||||||
|
'tag', 'bar',
|
||||||
|
null, ' { }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagStringNoQuotes',
|
||||||
|
'foo { font-family: hello world; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'font-family',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'variable-2', 'hello',
|
||||||
|
null, ' ',
|
||||||
|
'variable-2', 'world',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagStringDouble',
|
||||||
|
'foo { font-family: "hello world"; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'font-family',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'string', '"hello world"',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagStringSingle',
|
||||||
|
'foo { font-family: \'hello world\'; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'font-family',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'string', '\'hello world\'',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagColorKeyword',
|
||||||
|
'foo { color: black; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'color',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'keyword', 'black',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagColorHex3',
|
||||||
|
'foo { background: #fff; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'background',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'atom', '#fff',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagColorHex6',
|
||||||
|
'foo { background: #ffffff; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'background',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'atom', '#ffffff',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagColorHex4',
|
||||||
|
'foo { background: #ffff; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'background',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'atom error', '#ffff',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagColorHexInvalid',
|
||||||
|
'foo { background: #ffg; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'background',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'atom error', '#ffg',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagNegativeNumber',
|
||||||
|
'foo { margin: -5px; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'margin',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '-5px',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagPositiveNumber',
|
||||||
|
'foo { padding: 5px; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'padding',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '5px',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagVendor',
|
||||||
|
'foo { -foo-box-sizing: -foo-border-box; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'meta', '-foo-',
|
||||||
|
'property', 'box-sizing',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'meta', '-foo-',
|
||||||
|
'string-2', 'border-box',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagBogusProperty',
|
||||||
|
'foo { barhelloworld: 0; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property error', 'barhelloworld',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '0',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
MT.testMode(
|
||||||
|
'tagTwoProperties',
|
||||||
|
'foo { margin: 0; padding: 0; }',
|
||||||
|
[
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' { ',
|
||||||
|
'property', 'margin',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '0',
|
||||||
|
null, '; ',
|
||||||
|
'property', 'padding',
|
||||||
|
'operator', ':',
|
||||||
|
null, ' ',
|
||||||
|
'number', '0',
|
||||||
|
null, '; }'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
//
|
||||||
|
//MT.testMode(
|
||||||
|
// 'tagClass',
|
||||||
|
// '@media only screen and (min-width: 500px), print {foo.bar#hello { color: black !important; background: #f00; margin: -5px; padding: 5px; -foo-box-sizing: border-box; } /* world */}',
|
||||||
|
// [
|
||||||
|
// 'def', '@media',
|
||||||
|
// null, ' ',
|
||||||
|
// 'keyword', 'only',
|
||||||
|
// null, ' ',
|
||||||
|
// 'attribute', 'screen',
|
||||||
|
// null, ' ',
|
||||||
|
// 'operator', 'and',
|
||||||
|
// null, ' ',
|
||||||
|
// 'bracket', '(',
|
||||||
|
// 'property', 'min-width',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'number', '500px',
|
||||||
|
// 'bracket', ')',
|
||||||
|
// null, ', ',
|
||||||
|
// 'attribute', 'print',
|
||||||
|
// null, ' {',
|
||||||
|
// 'tag', 'foo',
|
||||||
|
// 'qualifier', '.bar',
|
||||||
|
// 'header', '#hello',
|
||||||
|
// null, ' { ',
|
||||||
|
// 'property', 'color',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'keyword', 'black',
|
||||||
|
// null, ' ',
|
||||||
|
// 'keyword', '!important',
|
||||||
|
// null, '; ',
|
||||||
|
// 'property', 'background',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'atom', '#f00',
|
||||||
|
// null, '; ',
|
||||||
|
// 'property', 'padding',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'number', '5px',
|
||||||
|
// null, '; ',
|
||||||
|
// 'property', 'margin',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'number', '-5px',
|
||||||
|
// null, '; ',
|
||||||
|
// 'meta', '-foo-',
|
||||||
|
// 'property', 'box-sizing',
|
||||||
|
// 'operator', ':',
|
||||||
|
// null, ' ',
|
||||||
|
// 'string-2', 'border-box',
|
||||||
|
// null, '; } ',
|
||||||
|
// 'comment', '/* world */',
|
||||||
|
// null, '}'
|
||||||
|
// ]
|
||||||
|
//);
|
32
codemirror-2.36/mode/diff/diff.js
Normal file
32
codemirror-2.36/mode/diff/diff.js
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
CodeMirror.defineMode("diff", function() {
|
||||||
|
|
||||||
|
var TOKEN_NAMES = {
|
||||||
|
'+': 'tag',
|
||||||
|
'-': 'string',
|
||||||
|
'@': '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");
|
105
codemirror-2.36/mode/diff/index.html
Normal file
105
codemirror-2.36/mode/diff/index.html
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Diff mode</title>
|
||||||
|
<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>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Diff mode</h1>
|
||||||
|
<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>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
203
codemirror-2.36/mode/ecl/ecl.js
Normal file
203
codemirror-2.36/mode/ecl/ecl.js
Normal file
|
@ -0,0 +1,203 @@
|
||||||
|
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";
|
||||||
|
}
|
||||||
|
|
||||||
|
function tokenAtString(stream, state) {
|
||||||
|
var next;
|
||||||
|
while ((next = stream.next()) != null) {
|
||||||
|
if (next == '"' && !stream.eat('"')) {
|
||||||
|
state.tokenize = null;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "string";
|
||||||
|
}
|
||||||
|
|
||||||
|
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 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 == "'") {
|
||||||
|
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 || multiLineStrings))
|
||||||
|
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");
|
42
codemirror-2.36/mode/ecl/index.html
Normal file
42
codemirror-2.36/mode/ecl/index.html
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>CodeMirror: ECL mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="ecl.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>.CodeMirror {border: 1px solid black;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: ECL mode</h1>
|
||||||
|
<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"), {
|
||||||
|
tabMode: "indent",
|
||||||
|
matchBrackets: true,
|
||||||
|
});
|
||||||
|
</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>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
463
codemirror-2.36/mode/erlang/erlang.js
Normal file
463
codemirror-2.36/mode/erlang/erlang.js
Normal file
|
@ -0,0 +1,463 @@
|
||||||
|
// block; "begin", "case", "fun", "if", "receive", "try": closed by "end"
|
||||||
|
// block internal; "after", "catch", "of"
|
||||||
|
// guard; "when", closed by "->"
|
||||||
|
// "->" opens a clause, closed by ";" or "."
|
||||||
|
// "<<" opens a binary, closed by ">>"
|
||||||
|
// "," appears in arglists, lists, tuples and terminates lines of code
|
||||||
|
// "." resets indentation to 0
|
||||||
|
// obsolete; "cond", "let", "query"
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-erlang", "erlang");
|
||||||
|
|
||||||
|
CodeMirror.defineMode("erlang", function(cmCfg, modeCfg) {
|
||||||
|
|
||||||
|
function rval(state,stream,type) {
|
||||||
|
// distinguish between "." as terminator and record field operator
|
||||||
|
if (type == "record") {
|
||||||
|
state.context = "record";
|
||||||
|
}else{
|
||||||
|
state.context = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// remember last significant bit on last line for indenting
|
||||||
|
if (type != "whitespace" && type != "comment") {
|
||||||
|
state.lastToken = stream.current();
|
||||||
|
}
|
||||||
|
// erlang -> CodeMirror tag
|
||||||
|
switch (type) {
|
||||||
|
case "atom": return "atom";
|
||||||
|
case "attribute": return "attribute";
|
||||||
|
case "builtin": return "builtin";
|
||||||
|
case "comment": return "comment";
|
||||||
|
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 "operator": return "operator";
|
||||||
|
case "record": return "bracket";
|
||||||
|
case "string": return "string";
|
||||||
|
case "type": return "def";
|
||||||
|
case "variable": return "variable";
|
||||||
|
case "error": return "error";
|
||||||
|
case "separator": return null;
|
||||||
|
case "open_paren": return null;
|
||||||
|
case "close_paren": return null;
|
||||||
|
default: return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var typeWords = [
|
||||||
|
"-type", "-spec", "-export_type", "-opaque"];
|
||||||
|
|
||||||
|
var keywordWords = [
|
||||||
|
"after","begin","catch","case","cond","end","fun","if",
|
||||||
|
"let","of","query","receive","try","when"];
|
||||||
|
|
||||||
|
var separatorWords = [
|
||||||
|
"->",";",":",".",","];
|
||||||
|
|
||||||
|
var operatorWords = [
|
||||||
|
"and","andalso","band","bnot","bor","bsl","bsr","bxor",
|
||||||
|
"div","not","or","orelse","rem","xor"];
|
||||||
|
|
||||||
|
var symbolWords = [
|
||||||
|
"+","-","*","/",">",">=","<","=<","=:=","==","=/=","/=","||","<-"];
|
||||||
|
|
||||||
|
var openParenWords = [
|
||||||
|
"<<","(","[","{"];
|
||||||
|
|
||||||
|
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"];
|
||||||
|
|
||||||
|
// ignored for indenting purposes
|
||||||
|
var ignoreWords = [
|
||||||
|
",", ":", "catch", "after", "of", "cond", "let", "query"];
|
||||||
|
|
||||||
|
|
||||||
|
var smallRE = /[a-z_]/;
|
||||||
|
var largeRE = /[A-Z_]/;
|
||||||
|
var digitRE = /[0-9]/;
|
||||||
|
var octitRE = /[0-7]/;
|
||||||
|
var anumRE = /[a-z_A-Z0-9]/;
|
||||||
|
var symbolRE = /[\+\-\*\/<>=\|:]/;
|
||||||
|
var openParenRE = /[<\(\[\{]/;
|
||||||
|
var closeParenRE = /[>\)\]\}]/;
|
||||||
|
var sepRE = /[\->\.,:;]/;
|
||||||
|
|
||||||
|
function isMember(element,list) {
|
||||||
|
return (-1 < list.indexOf(element));
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPrev(stream,string) {
|
||||||
|
var start = stream.start;
|
||||||
|
var len = string.length;
|
||||||
|
if (len <= start) {
|
||||||
|
var word = stream.string.slice(start-len,start);
|
||||||
|
return word == string;
|
||||||
|
}else{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function tokenize(stream, state) {
|
||||||
|
if (stream.eatSpace()) {
|
||||||
|
return rval(state,stream,"whitespace");
|
||||||
|
}
|
||||||
|
|
||||||
|
// attributes and type specs
|
||||||
|
if ((peekToken(state).token == "" || peekToken(state).token == ".") &&
|
||||||
|
stream.peek() == '-') {
|
||||||
|
stream.next();
|
||||||
|
if (stream.eat(smallRE) && stream.eatWhile(anumRE)) {
|
||||||
|
if (isMember(stream.current(),typeWords)) {
|
||||||
|
return rval(state,stream,"type");
|
||||||
|
}else{
|
||||||
|
return rval(state,stream,"attribute");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stream.backUp(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
var ch = stream.next();
|
||||||
|
|
||||||
|
// comment
|
||||||
|
if (ch == '%') {
|
||||||
|
stream.skipToEnd();
|
||||||
|
return rval(state,stream,"comment");
|
||||||
|
}
|
||||||
|
|
||||||
|
// macro
|
||||||
|
if (ch == '?') {
|
||||||
|
stream.eatWhile(anumRE);
|
||||||
|
return rval(state,stream,"macro");
|
||||||
|
}
|
||||||
|
|
||||||
|
// record
|
||||||
|
if ( ch == "#") {
|
||||||
|
stream.eatWhile(anumRE);
|
||||||
|
return rval(state,stream,"record");
|
||||||
|
}
|
||||||
|
|
||||||
|
// char
|
||||||
|
if ( ch == "$") {
|
||||||
|
if (stream.next() == "\\") {
|
||||||
|
if (!stream.eatWhile(octitRE)) {
|
||||||
|
stream.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return rval(state,stream,"string");
|
||||||
|
}
|
||||||
|
|
||||||
|
// quoted atom
|
||||||
|
if (ch == '\'') {
|
||||||
|
if (singleQuote(stream)) {
|
||||||
|
return rval(state,stream,"atom");
|
||||||
|
}else{
|
||||||
|
return rval(state,stream,"error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// string
|
||||||
|
if (ch == '"') {
|
||||||
|
if (doubleQuote(stream)) {
|
||||||
|
return rval(state,stream,"string");
|
||||||
|
}else{
|
||||||
|
return rval(state,stream,"error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// variable
|
||||||
|
if (largeRE.test(ch)) {
|
||||||
|
stream.eatWhile(anumRE);
|
||||||
|
return rval(state,stream,"variable");
|
||||||
|
}
|
||||||
|
|
||||||
|
// atom/keyword/BIF/function
|
||||||
|
if (smallRE.test(ch)) {
|
||||||
|
stream.eatWhile(anumRE);
|
||||||
|
|
||||||
|
if (stream.peek() == "/") {
|
||||||
|
stream.next();
|
||||||
|
if (stream.eatWhile(digitRE)) {
|
||||||
|
return rval(state,stream,"fun"); // f/0 style fun
|
||||||
|
}else{
|
||||||
|
stream.backUp(1);
|
||||||
|
return rval(state,stream,"atom");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var w = stream.current();
|
||||||
|
|
||||||
|
if (isMember(w,keywordWords)) {
|
||||||
|
pushToken(state,stream);
|
||||||
|
return rval(state,stream,"keyword");
|
||||||
|
}
|
||||||
|
if (stream.peek() == "(") {
|
||||||
|
// 'put' and 'erlang:put' are bifs, 'foo:put' is not
|
||||||
|
if (isMember(w,bifWords) &&
|
||||||
|
(!isPrev(stream,":") || isPrev(stream,"erlang:"))) {
|
||||||
|
return rval(state,stream,"builtin");
|
||||||
|
}else{
|
||||||
|
return rval(state,stream,"function");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (isMember(w,guardWords)) {
|
||||||
|
return rval(state,stream,"guard");
|
||||||
|
}
|
||||||
|
if (isMember(w,operatorWords)) {
|
||||||
|
return rval(state,stream,"operator");
|
||||||
|
}
|
||||||
|
if (stream.peek() == ":") {
|
||||||
|
if (w == "erlang") {
|
||||||
|
return rval(state,stream,"builtin");
|
||||||
|
} else {
|
||||||
|
return rval(state,stream,"function");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return rval(state,stream,"atom");
|
||||||
|
}
|
||||||
|
|
||||||
|
// number
|
||||||
|
if (digitRE.test(ch)) {
|
||||||
|
stream.eatWhile(digitRE);
|
||||||
|
if (stream.eat('#')) {
|
||||||
|
stream.eatWhile(digitRE); // 16#10 style integer
|
||||||
|
} else {
|
||||||
|
if (stream.eat('.')) { // float
|
||||||
|
stream.eatWhile(digitRE);
|
||||||
|
}
|
||||||
|
if (stream.eat(/[eE]/)) {
|
||||||
|
stream.eat(/[-+]/); // float with exponent
|
||||||
|
stream.eatWhile(digitRE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return rval(state,stream,"number"); // normal integer
|
||||||
|
}
|
||||||
|
|
||||||
|
// open parens
|
||||||
|
if (nongreedy(stream,openParenRE,openParenWords)) {
|
||||||
|
pushToken(state,stream);
|
||||||
|
return rval(state,stream,"open_paren");
|
||||||
|
}
|
||||||
|
|
||||||
|
// close parens
|
||||||
|
if (nongreedy(stream,closeParenRE,closeParenWords)) {
|
||||||
|
pushToken(state,stream);
|
||||||
|
return rval(state,stream,"close_paren");
|
||||||
|
}
|
||||||
|
|
||||||
|
// separators
|
||||||
|
if (greedy(stream,sepRE,separatorWords)) {
|
||||||
|
// distinguish between "." as terminator and record field operator
|
||||||
|
if (state.context == false) {
|
||||||
|
pushToken(state,stream);
|
||||||
|
}
|
||||||
|
return rval(state,stream,"separator");
|
||||||
|
}
|
||||||
|
|
||||||
|
// operators
|
||||||
|
if (greedy(stream,symbolRE,symbolWords)) {
|
||||||
|
return rval(state,stream,"operator");
|
||||||
|
}
|
||||||
|
|
||||||
|
return rval(state,stream,null);
|
||||||
|
}
|
||||||
|
|
||||||
|
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 (isMember(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 (isMember(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 Token(stream) {
|
||||||
|
this.token = stream ? stream.current() : "";
|
||||||
|
this.column = stream ? stream.column() : 0;
|
||||||
|
this.indent = stream ? stream.indentation() : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
function myIndent(state,textAfter) {
|
||||||
|
var indent = cmCfg.indentUnit;
|
||||||
|
var outdentWords = ["after","catch"];
|
||||||
|
var token = (peekToken(state)).token;
|
||||||
|
var wordAfter = takewhile(textAfter,/[^a-z]/);
|
||||||
|
|
||||||
|
if (isMember(token,openParenWords)) {
|
||||||
|
return (peekToken(state)).column+token.length;
|
||||||
|
}else if (token == "." || token == ""){
|
||||||
|
return 0;
|
||||||
|
}else if (token == "->") {
|
||||||
|
if (wordAfter == "end") {
|
||||||
|
return peekToken(state,2).column;
|
||||||
|
}else if (peekToken(state,2).token == "fun") {
|
||||||
|
return peekToken(state,2).column+indent;
|
||||||
|
}else{
|
||||||
|
return (peekToken(state)).indent+indent;
|
||||||
|
}
|
||||||
|
}else if (isMember(wordAfter,outdentWords)) {
|
||||||
|
return (peekToken(state)).indent;
|
||||||
|
}else{
|
||||||
|
return (peekToken(state)).column+indent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function takewhile(str,re) {
|
||||||
|
var m = str.match(re);
|
||||||
|
return m ? str.slice(0,m.index) : str;
|
||||||
|
}
|
||||||
|
|
||||||
|
function popToken(state) {
|
||||||
|
return state.tokenStack.pop();
|
||||||
|
}
|
||||||
|
|
||||||
|
function peekToken(state,depth) {
|
||||||
|
var len = state.tokenStack.length;
|
||||||
|
var dep = (depth ? depth : 1);
|
||||||
|
if (len < dep) {
|
||||||
|
return new Token;
|
||||||
|
}else{
|
||||||
|
return state.tokenStack[len-dep];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function pushToken(state,stream) {
|
||||||
|
var token = stream.current();
|
||||||
|
var prev_token = peekToken(state).token;
|
||||||
|
if (isMember(token,ignoreWords)) {
|
||||||
|
return false;
|
||||||
|
}else if (drop_both(prev_token,token)) {
|
||||||
|
popToken(state);
|
||||||
|
return false;
|
||||||
|
}else if (drop_first(prev_token,token)) {
|
||||||
|
popToken(state);
|
||||||
|
return pushToken(state,stream);
|
||||||
|
}else{
|
||||||
|
state.tokenStack.push(new Token(stream));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function drop_first(open, close) {
|
||||||
|
switch (open+" "+close) {
|
||||||
|
case "when ->": return true;
|
||||||
|
case "-> end": return true;
|
||||||
|
case "-> .": return true;
|
||||||
|
case ". .": return true;
|
||||||
|
default: return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function drop_both(open, close) {
|
||||||
|
switch (open+" "+close) {
|
||||||
|
case "( )": return true;
|
||||||
|
case "[ ]": return true;
|
||||||
|
case "{ }": return true;
|
||||||
|
case "<< >>": return true;
|
||||||
|
case "begin end": return true;
|
||||||
|
case "case end": return true;
|
||||||
|
case "fun end": return true;
|
||||||
|
case "if end": return true;
|
||||||
|
case "receive end": return true;
|
||||||
|
case "try end": return true;
|
||||||
|
case "-> ;": return true;
|
||||||
|
default: return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState:
|
||||||
|
function() {
|
||||||
|
return {tokenStack: [],
|
||||||
|
context: false,
|
||||||
|
lastToken: null};
|
||||||
|
},
|
||||||
|
|
||||||
|
token:
|
||||||
|
function(stream, state) {
|
||||||
|
return tokenize(stream, state);
|
||||||
|
},
|
||||||
|
|
||||||
|
indent:
|
||||||
|
function(state, textAfter) {
|
||||||
|
// console.log(state.tokenStack);
|
||||||
|
return myIndent(state,textAfter);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
63
codemirror-2.36/mode/erlang/index.html
Normal file
63
codemirror-2.36/mode/erlang/index.html
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Erlang mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="erlang.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../theme/erlang-dark.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Erlang mode</h1>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
</body>
|
||||||
|
</html>
|
94
codemirror-2.36/mode/gfm/gfm.js
Normal file
94
codemirror-2.36/mode/gfm/gfm.js
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
CodeMirror.defineMode("gfm", function(config, parserConfig) {
|
||||||
|
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) {
|
||||||
|
// 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(stream.match(/^(?:[a-zA-Z0-9\-_]+\/)?(?:[a-zA-Z0-9\-_]+@)?(?:[a-f0-9]{7,40}\b)/)) {
|
||||||
|
// User/Project@SHA
|
||||||
|
// User@SHA
|
||||||
|
// SHA
|
||||||
|
return "link";
|
||||||
|
} else if (stream.match(/^(?:[a-zA-Z0-9\-_]+\/)?(?:[a-zA-Z0-9\-_]+)?#[0-9]+\b/)) {
|
||||||
|
// User/Project#Num
|
||||||
|
// User#Num
|
||||||
|
// #Num
|
||||||
|
return "link";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (stream.match(/^((?:[a-z][\w-]+:(?:\/{1,3}|[a-z0-9%])|www\d{0,3}[.]|[a-z0-9.\-]+[.][a-z]{2,4}\/)(?:[^\s()<>]+|\(([^\s()<>]+|(\([^\s()<>]+\)))*\))+(?:\(([^\s()<>]+|(\([^\s()<>]+\)))*\)|[^\s`!()\[\]{};:'".,<>?«»“”‘’]))/i)) {
|
||||||
|
// URLs
|
||||||
|
// Taken from http://daringfireball.net/2010/07/improved_regex_for_matching_urls
|
||||||
|
return "link";
|
||||||
|
}
|
||||||
|
stream.next();
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
blankLine: blankLine
|
||||||
|
};
|
||||||
|
CodeMirror.defineMIME("gfmBase", {
|
||||||
|
name: "markdown",
|
||||||
|
underscoresBreakWords: false,
|
||||||
|
fencedCodeBlocks: true
|
||||||
|
});
|
||||||
|
return CodeMirror.overlayMode(CodeMirror.getMode(config, "gfmBase"), gfmOverlay);
|
||||||
|
}, "markdown");
|
71
codemirror-2.36/mode/gfm/index.html
Normal file
71
codemirror-2.36/mode/gfm/index.html
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: GFM mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="../../lib/util/overlay.js"></script>
|
||||||
|
<script src="../xml/xml.js"></script>
|
||||||
|
<script src="../markdown/markdown.js"></script>
|
||||||
|
<script src="gfm.js"></script>
|
||||||
|
|
||||||
|
<!-- Code block highlighting modes -->
|
||||||
|
<script src="../javascript/javascript.js"></script>
|
||||||
|
<script src="../css/css.js"></script>
|
||||||
|
<script src="../htmlmixed/htmlmixed.js"></script>
|
||||||
|
<script src="../clike/clike.js"></script>
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="../markdown/markdown.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: GFM mode</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
GitHub Flavored Markdown
|
||||||
|
========================
|
||||||
|
|
||||||
|
Everything from markdown plus GFM features:
|
||||||
|
|
||||||
|
## URL autolinking
|
||||||
|
|
||||||
|
Underscores_are_allowed_between_words.
|
||||||
|
|
||||||
|
## Fenced code blocks (and syntax highlighting)
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
for (var i = 0; i < items.length; i++) {
|
||||||
|
console.log(items[i], i); // log them
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 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,
|
||||||
|
matchBrackets: 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>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
225
codemirror-2.36/mode/gfm/test.js
Normal file
225
codemirror-2.36/mode/gfm/test.js
Normal file
|
@ -0,0 +1,225 @@
|
||||||
|
// Initiate ModeTest and set defaults
|
||||||
|
var MT = ModeTest;
|
||||||
|
MT.modeName = 'gfm';
|
||||||
|
MT.modeOptions = {};
|
||||||
|
|
||||||
|
// Emphasis characters within a word
|
||||||
|
MT.testMode(
|
||||||
|
'emInWordAsterisk',
|
||||||
|
'foo*bar*hello',
|
||||||
|
[
|
||||||
|
null, 'foo',
|
||||||
|
'em', '*bar*',
|
||||||
|
null, 'hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
MT.testMode(
|
||||||
|
'emInWordUnderscore',
|
||||||
|
'foo_bar_hello',
|
||||||
|
[
|
||||||
|
null, 'foo_bar_hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
MT.testMode(
|
||||||
|
'emStrongUnderscore',
|
||||||
|
'___foo___ bar',
|
||||||
|
[
|
||||||
|
'strong', '__',
|
||||||
|
'emstrong', '_foo__',
|
||||||
|
'em', '_',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Fenced code blocks
|
||||||
|
MT.testMode(
|
||||||
|
'fencedCodeBlocks',
|
||||||
|
'```\nfoo\n\n```\nbar',
|
||||||
|
[
|
||||||
|
'comment', '```',
|
||||||
|
'comment', 'foo',
|
||||||
|
'comment', '```',
|
||||||
|
null, 'bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// Fenced code block mode switching
|
||||||
|
MT.testMode(
|
||||||
|
'fencedCodeBlockModeSwitching',
|
||||||
|
'```javascript\nfoo\n\n```\nbar',
|
||||||
|
[
|
||||||
|
'comment', '```javascript',
|
||||||
|
'variable', 'foo',
|
||||||
|
'comment', '```',
|
||||||
|
null, 'bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// SHA
|
||||||
|
MT.testMode(
|
||||||
|
'SHA',
|
||||||
|
'foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// GitHub highlights hashes 7-40 chars in length
|
||||||
|
MT.testMode(
|
||||||
|
'shortSHA',
|
||||||
|
'foo be6a8cc bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'be6a8cc',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// Invalid SHAs
|
||||||
|
//
|
||||||
|
// GitHub does not highlight hashes shorter than 7 chars
|
||||||
|
MT.testMode(
|
||||||
|
'tooShortSHA',
|
||||||
|
'foo be6a8c bar',
|
||||||
|
[
|
||||||
|
null, 'foo be6a8c bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// GitHub does not highlight hashes longer than 40 chars
|
||||||
|
MT.testMode(
|
||||||
|
'longSHA',
|
||||||
|
'foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd22 bar',
|
||||||
|
[
|
||||||
|
null, 'foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd22 bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
MT.testMode(
|
||||||
|
'badSHA',
|
||||||
|
'foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cg2 bar',
|
||||||
|
[
|
||||||
|
null, 'foo be6a8cc1c1ecfe9489fb51e4869af15a13fc2cg2 bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// User@SHA
|
||||||
|
MT.testMode(
|
||||||
|
'userSHA',
|
||||||
|
'foo bar@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 hello',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'bar@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2',
|
||||||
|
null, ' hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// User/Project@SHA
|
||||||
|
MT.testMode(
|
||||||
|
'userProjectSHA',
|
||||||
|
'foo bar/hello@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2 world',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'bar/hello@be6a8cc1c1ecfe9489fb51e4869af15a13fc2cd2',
|
||||||
|
null, ' world'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// #Num
|
||||||
|
MT.testMode(
|
||||||
|
'num',
|
||||||
|
'foo #1 bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', '#1',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// bad #Num
|
||||||
|
MT.testMode(
|
||||||
|
'badNum',
|
||||||
|
'foo #1bar hello',
|
||||||
|
[
|
||||||
|
null, 'foo #1bar hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// User#Num
|
||||||
|
MT.testMode(
|
||||||
|
'userNum',
|
||||||
|
'foo bar#1 hello',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'bar#1',
|
||||||
|
null, ' hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// User/Project#Num
|
||||||
|
MT.testMode(
|
||||||
|
'userProjectNum',
|
||||||
|
'foo bar/hello#1 world',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'bar/hello#1',
|
||||||
|
null, ' world'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Vanilla links
|
||||||
|
MT.testMode(
|
||||||
|
'vanillaLink',
|
||||||
|
'foo http://www.example.com/ bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'http://www.example.com/',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
MT.testMode(
|
||||||
|
'vanillaLinkPunctuation',
|
||||||
|
'foo http://www.example.com/. bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'http://www.example.com/',
|
||||||
|
null, '. bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
MT.testMode(
|
||||||
|
'vanillaLinkExtension',
|
||||||
|
'foo http://www.example.com/index.html bar',
|
||||||
|
[
|
||||||
|
null, 'foo ',
|
||||||
|
'link', 'http://www.example.com/index.html',
|
||||||
|
null, ' bar'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// Not a link
|
||||||
|
MT.testMode(
|
||||||
|
'notALink',
|
||||||
|
'```css\nfoo {color:black;}\n```http://www.example.com/',
|
||||||
|
[
|
||||||
|
'comment', '```css',
|
||||||
|
'tag', 'foo',
|
||||||
|
null, ' {',
|
||||||
|
'property', 'color',
|
||||||
|
'operator', ':',
|
||||||
|
'keyword', 'black',
|
||||||
|
null, ';}',
|
||||||
|
'comment', '```',
|
||||||
|
'link', 'http://www.example.com/'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// Not a link
|
||||||
|
MT.testMode(
|
||||||
|
'notALink',
|
||||||
|
'``foo `bar` http://www.example.com/`` hello',
|
||||||
|
[
|
||||||
|
'comment', '``foo `bar` http://www.example.com/``',
|
||||||
|
null, ' hello'
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// Not a link
|
||||||
|
MT.testMode(
|
||||||
|
'notALink',
|
||||||
|
'`foo\nhttp://www.example.com/\n`foo\n\nhttp://www.example.com/',
|
||||||
|
[
|
||||||
|
'comment', '`foo',
|
||||||
|
'link', 'http://www.example.com/',
|
||||||
|
'comment', '`foo',
|
||||||
|
'link', 'http://www.example.com/'
|
||||||
|
]
|
||||||
|
);
|
170
codemirror-2.36/mode/go/go.js
Normal file
170
codemirror-2.36/mode/go/go.js
Normal file
|
@ -0,0 +1,170 @@
|
||||||
|
CodeMirror.defineMode("go", function(config, parserConfig) {
|
||||||
|
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
|
||||||
|
};
|
||||||
|
|
||||||
|
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 blockKeywords = {
|
||||||
|
"else":true, "for":true, "func":true, "if":true, "interface":true,
|
||||||
|
"select":true, "struct":true, "switch":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\$_]/);
|
||||||
|
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 && 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) {
|
||||||
|
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: "{}:"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-go", "go");
|
73
codemirror-2.36/mode/go/index.html
Normal file
73
codemirror-2.36/mode/go/index.html
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Go mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<link rel="stylesheet" href="../../theme/elegant.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="go.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>.CodeMirror {border:1px solid #999; background:#ffc}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Go mode</h1>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
</body>
|
||||||
|
</html>
|
210
codemirror-2.36/mode/groovy/groovy.js
Normal file
210
codemirror-2.36/mode/groovy/groovy.js
Normal file
|
@ -0,0 +1,210 @@
|
||||||
|
CodeMirror.defineMode("groovy", function(config, parserConfig) {
|
||||||
|
function words(str) {
|
||||||
|
var obj = {}, words = str.split(" ");
|
||||||
|
for (var i = 0; i < words.length; ++i) obj[words[i]] = true;
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
var 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 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)) {
|
||||||
|
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";
|
||||||
|
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) {
|
||||||
|
return !last || last == "operator" || last == "->" || /[\.\[\{\(,;:]/.test(last) ||
|
||||||
|
last == "newstatement" || last == "keyword" || last == "proplabel";
|
||||||
|
}
|
||||||
|
|
||||||
|
function Context(indented, column, type, align, prev) {
|
||||||
|
this.indented = indented;
|
||||||
|
this.column = column;
|
||||||
|
this.type = type;
|
||||||
|
this.align = align;
|
||||||
|
this.prev = prev;
|
||||||
|
}
|
||||||
|
function pushContext(state, col, type) {
|
||||||
|
return state.context = new Context(state.indented, col, type, null, state.context);
|
||||||
|
}
|
||||||
|
function popContext(state) {
|
||||||
|
var t = state.context.type;
|
||||||
|
if (t == ")" || t == "]" || t == "}")
|
||||||
|
state.indented = state.context.indented;
|
||||||
|
return state.context = state.context.prev;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState: function(basecolumn) {
|
||||||
|
return {
|
||||||
|
tokenize: [tokenBase],
|
||||||
|
context: new Context((basecolumn || 0) - config.indentUnit, 0, "top", false),
|
||||||
|
indented: 0,
|
||||||
|
startOfLine: true,
|
||||||
|
lastToken: null
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
token: function(stream, state) {
|
||||||
|
var ctx = state.context;
|
||||||
|
if (stream.sol()) {
|
||||||
|
if (ctx.align == null) ctx.align = false;
|
||||||
|
state.indented = stream.indentation();
|
||||||
|
state.startOfLine = true;
|
||||||
|
// Automatic semicolon insertion
|
||||||
|
if (ctx.type == "statement" && !expectExpression(state.lastToken)) {
|
||||||
|
popContext(state); ctx = state.context;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (stream.eatSpace()) return null;
|
||||||
|
curPunc = null;
|
||||||
|
var style = state.tokenize[state.tokenize.length-1](stream, state);
|
||||||
|
if (style == "comment") return style;
|
||||||
|
if (ctx.align == null) ctx.align = true;
|
||||||
|
|
||||||
|
if ((curPunc == ";" || curPunc == ":") && ctx.type == "statement") popContext(state);
|
||||||
|
// 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)) 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: "{}"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-groovy", "groovy");
|
72
codemirror-2.36/mode/groovy/index.html
Normal file
72
codemirror-2.36/mode/groovy/index.html
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Groovy mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="groovy.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>.CodeMirror {border-top: 1px solid #500; border-bottom: 1px solid #500;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Groovy mode</h1>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
</body>
|
||||||
|
</html>
|
242
codemirror-2.36/mode/haskell/haskell.js
Normal file
242
codemirror-2.36/mode/haskell/haskell.js
Normal file
|
@ -0,0 +1,242 @@
|
||||||
|
CodeMirror.defineMode("haskell", function(cmCfg, modeCfg) {
|
||||||
|
|
||||||
|
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 = /[-!#$%&*+.\/<=>?@\\^|~:]/;
|
||||||
|
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.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";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
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");
|
||||||
|
|
||||||
|
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 (w in wellKnownWords) ? wellKnownWords[w] : t;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-haskell", "haskell");
|
61
codemirror-2.36/mode/haskell/index.html
Normal file
61
codemirror-2.36/mode/haskell/index.html
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Haskell mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="haskell.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../theme/elegant.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Haskell mode</h1>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
</body>
|
||||||
|
</html>
|
429
codemirror-2.36/mode/haxe/haxe.js
Normal file
429
codemirror-2.36/mode/haxe/haxe.js
Normal file
|
@ -0,0 +1,429 @@
|
||||||
|
CodeMirror.defineMode("haxe", function(config, parserConfig) {
|
||||||
|
var indentUnit = config.indentUnit;
|
||||||
|
|
||||||
|
// Tokenizer
|
||||||
|
|
||||||
|
var keywords = function(){
|
||||||
|
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");
|
||||||
|
return {
|
||||||
|
"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, "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 nextUntilUnescaped(stream, end) {
|
||||||
|
var escaped = false, next;
|
||||||
|
while ((next = stream.next()) != null) {
|
||||||
|
if (next == end && !escaped)
|
||||||
|
return false;
|
||||||
|
escaped = !escaped && next == "\\";
|
||||||
|
}
|
||||||
|
return escaped;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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(/\//))) {
|
||||||
|
nextUntilUnescaped(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 (!nextUntilUnescaped(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 register(varname) {
|
||||||
|
var state = cx.state;
|
||||||
|
if (state.context) {
|
||||||
|
cx.marked = "def";
|
||||||
|
for (var v = state.localVars; v; v = v.next)
|
||||||
|
if (v.name == varname) return;
|
||||||
|
state.localVars = {name: varname, next: state.localVars};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
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) {
|
||||||
|
return function expecting(type) {
|
||||||
|
if (type == wanted) return cont();
|
||||||
|
else if (wanted == ";") return pass();
|
||||||
|
else return cont(arguments.callee);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
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 == "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(expression, "]"), 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, value) {
|
||||||
|
if (type == "attribute") return cont(maybeattribute);
|
||||||
|
if (type == "function") return cont(functiondef);
|
||||||
|
if (type == "var") return cont(vardef1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function metadef(type, value) {
|
||||||
|
if(type == ":") return cont(metadef);
|
||||||
|
if(type == "variable") return cont(metadef);
|
||||||
|
if(type == "(") return cont(pushlex(")"), comasep(metaargs, ")"), poplex, statement);
|
||||||
|
}
|
||||||
|
function metaargs(type, value) {
|
||||||
|
if(typ == "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 == ".") return cont(importdef);
|
||||||
|
}
|
||||||
|
|
||||||
|
function typedef (type, value)
|
||||||
|
{
|
||||||
|
if(type == "variable" && /[A-Z]/.test(value.charAt(0))) { registerimport(value); 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 commaSeparated(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(pushlex(")"), pushcontext, forin, expression, poplex, statement, popcontext);
|
||||||
|
}
|
||||||
|
function forin(type, value) {
|
||||||
|
if (value == "in") return cont();
|
||||||
|
}
|
||||||
|
function functiondef(type, value) {
|
||||||
|
if (type == "variable") {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, value) {
|
||||||
|
if(type == ":") return cont(typestring);
|
||||||
|
}
|
||||||
|
function typestring(type, value) {
|
||||||
|
if(type == "type") return cont();
|
||||||
|
if(type == "variable") return cont();
|
||||||
|
if(type == "{") return cont(pushlex("}"), commasep(typeprop, "}"), poplex);
|
||||||
|
}
|
||||||
|
function typeprop(type, value) {
|
||||||
|
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"];
|
||||||
|
return {
|
||||||
|
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
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
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: "{}"
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/x-haxe", "haxe");
|
91
codemirror-2.36/mode/haxe/index.html
Normal file
91
codemirror-2.36/mode/haxe/index.html
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Haxe mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="haxe.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Haxe mode</h1>
|
||||||
|
|
||||||
|
<div><textarea id="code" name="code">
|
||||||
|
import one.two.Three;
|
||||||
|
|
||||||
|
@attr("test")
|
||||||
|
class Foo<T> extends Three
|
||||||
|
{
|
||||||
|
public function new()
|
||||||
|
{
|
||||||
|
noFoo = 12;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static inline function doFoo(obj:{k:Int, l:Float}):Int
|
||||||
|
{
|
||||||
|
for(i in 0...10)
|
||||||
|
{
|
||||||
|
obj.k++;
|
||||||
|
trace(i);
|
||||||
|
var var1 = new Array();
|
||||||
|
if(var1.length > 1)
|
||||||
|
throw "Error";
|
||||||
|
}
|
||||||
|
// The following line should not be colored, the variable is scoped out
|
||||||
|
var1;
|
||||||
|
/* Multi line
|
||||||
|
* Comment test
|
||||||
|
*/
|
||||||
|
return obj.k;
|
||||||
|
}
|
||||||
|
private function bar():Void
|
||||||
|
{
|
||||||
|
#if flash
|
||||||
|
var t1:String = "1.21";
|
||||||
|
#end
|
||||||
|
try {
|
||||||
|
doFoo({k:3, l:1.2});
|
||||||
|
}
|
||||||
|
catch (e : String) {
|
||||||
|
trace(e);
|
||||||
|
}
|
||||||
|
var t2:Float = cast(3.2);
|
||||||
|
var t3:haxe.Timer = new haxe.Timer();
|
||||||
|
var t4 = {k:Std.int(t2), l:Std.parseFloat(t1)};
|
||||||
|
var t5 = ~/123+.*$/i;
|
||||||
|
doFoo(t4);
|
||||||
|
untyped t1 = 4;
|
||||||
|
bob = new Foo<Int>
|
||||||
|
}
|
||||||
|
public var okFoo(default, never):Float;
|
||||||
|
var noFoo(getFoo, null):Int;
|
||||||
|
function getFoo():Int {
|
||||||
|
return noFoo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public var three:Int;
|
||||||
|
}
|
||||||
|
enum Color
|
||||||
|
{
|
||||||
|
red;
|
||||||
|
green;
|
||||||
|
blue;
|
||||||
|
grey( v : Int );
|
||||||
|
rgb (r:Int,g:Int,b:Int);
|
||||||
|
}
|
||||||
|
</textarea></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
indentUnit: 4,
|
||||||
|
indentWithTabs: true
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/x-haxe</code>.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
73
codemirror-2.36/mode/htmlembedded/htmlembedded.js
Normal file
73
codemirror-2.36/mode/htmlembedded/htmlembedded.js
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
CodeMirror.defineMode("htmlembedded", function(config, parserConfig) {
|
||||||
|
|
||||||
|
//config settings
|
||||||
|
var scriptStartRegex = parserConfig.scriptStartRegex || /^<%/i,
|
||||||
|
scriptEndRegex = parserConfig.scriptEndRegex || /^%>/i;
|
||||||
|
|
||||||
|
//inner modes
|
||||||
|
var scriptingMode, htmlMixedMode;
|
||||||
|
|
||||||
|
//tokenizer when in html mode
|
||||||
|
function htmlDispatch(stream, state) {
|
||||||
|
if (stream.match(scriptStartRegex, false)) {
|
||||||
|
state.token=scriptingDispatch;
|
||||||
|
return scriptingMode.token(stream, state.scriptState);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return htmlMixedMode.token(stream, state.htmlState);
|
||||||
|
}
|
||||||
|
|
||||||
|
//tokenizer when in scripting mode
|
||||||
|
function scriptingDispatch(stream, state) {
|
||||||
|
if (stream.match(scriptEndRegex, false)) {
|
||||||
|
state.token=htmlDispatch;
|
||||||
|
return htmlMixedMode.token(stream, state.htmlState);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return scriptingMode.token(stream, state.scriptState);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState: function() {
|
||||||
|
scriptingMode = scriptingMode || CodeMirror.getMode(config, parserConfig.scriptingModeSpec);
|
||||||
|
htmlMixedMode = htmlMixedMode || CodeMirror.getMode(config, "htmlmixed");
|
||||||
|
return {
|
||||||
|
token : parserConfig.startOpen ? scriptingDispatch : htmlDispatch,
|
||||||
|
htmlState : htmlMixedMode.startState(),
|
||||||
|
scriptState : scriptingMode.startState()
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
token: function(stream, state) {
|
||||||
|
return state.token(stream, state);
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
if (state.token == htmlDispatch)
|
||||||
|
return htmlMixedMode.indent(state.htmlState, textAfter);
|
||||||
|
else
|
||||||
|
return scriptingMode.indent(state.scriptState, textAfter);
|
||||||
|
},
|
||||||
|
|
||||||
|
copyState: function(state) {
|
||||||
|
return {
|
||||||
|
token : state.token,
|
||||||
|
htmlState : CodeMirror.copyState(htmlMixedMode, state.htmlState),
|
||||||
|
scriptState : CodeMirror.copyState(scriptingMode, state.scriptState)
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
electricChars: "/{}:",
|
||||||
|
|
||||||
|
innerMode: function(state) {
|
||||||
|
if (state.token == scriptingDispatch) return {state: state.scriptState, mode: scriptingMode};
|
||||||
|
else return {state: state.htmlState, mode: htmlMixedMode};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, "htmlmixed");
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("application/x-ejs", { name: "htmlembedded", scriptingModeSpec:"javascript"});
|
||||||
|
CodeMirror.defineMIME("application/x-aspx", { name: "htmlembedded", scriptingModeSpec:"text/x-csharp"});
|
||||||
|
CodeMirror.defineMIME("application/x-jsp", { name: "htmlembedded", scriptingModeSpec:"text/x-java"});
|
||||||
|
CodeMirror.defineMIME("application/x-erb", { name: "htmlembedded", scriptingModeSpec:"ruby"});
|
50
codemirror-2.36/mode/htmlembedded/index.html
Normal file
50
codemirror-2.36/mode/htmlembedded/index.html
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Html Embedded Scripts mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="../xml/xml.js"></script>
|
||||||
|
<script src="../javascript/javascript.js"></script>
|
||||||
|
<script src="../css/css.js"></script>
|
||||||
|
<script src="../htmlmixed/htmlmixed.js"></script>
|
||||||
|
<script src="htmlembedded.js"></script>
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Html Embedded Scripts mode</h1>
|
||||||
|
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<%
|
||||||
|
function hello(who) {
|
||||||
|
return "Hello " + who;
|
||||||
|
}
|
||||||
|
%>
|
||||||
|
This is an example of EJS (embedded javascript)
|
||||||
|
<p>The program says <%= hello("world") %>.</p>
|
||||||
|
<script>
|
||||||
|
alert("And here is some normal JS code"); // also colored
|
||||||
|
</script>
|
||||||
|
</textarea></form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
mode: "application/x-ejs",
|
||||||
|
indentUnit: 4,
|
||||||
|
indentWithTabs: true,
|
||||||
|
enterMode: "keep",
|
||||||
|
tabMode: "shift"
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>Mode for html embedded scripts like JSP and ASP.NET. Depends on HtmlMixed which in turn depends on
|
||||||
|
JavaScript, CSS and XML.<br />Other dependancies include those of the scriping language chosen.</p>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>application/x-aspx</code> (ASP.NET),
|
||||||
|
<code>application/x-ejs</code> (Embedded Javascript), <code>application/x-jsp</code> (JavaServer Pages)</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
84
codemirror-2.36/mode/htmlmixed/htmlmixed.js
Normal file
84
codemirror-2.36/mode/htmlmixed/htmlmixed.js
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
CodeMirror.defineMode("htmlmixed", function(config) {
|
||||||
|
var htmlMode = CodeMirror.getMode(config, {name: "xml", htmlMode: true});
|
||||||
|
var jsMode = CodeMirror.getMode(config, "javascript");
|
||||||
|
var cssMode = CodeMirror.getMode(config, "css");
|
||||||
|
|
||||||
|
function html(stream, state) {
|
||||||
|
var style = htmlMode.token(stream, state.htmlState);
|
||||||
|
if (style == "tag" && stream.current() == ">" && state.htmlState.context) {
|
||||||
|
if (/^script$/i.test(state.htmlState.context.tagName)) {
|
||||||
|
state.token = javascript;
|
||||||
|
state.localState = jsMode.startState(htmlMode.indent(state.htmlState, ""));
|
||||||
|
}
|
||||||
|
else if (/^style$/i.test(state.htmlState.context.tagName)) {
|
||||||
|
state.token = css;
|
||||||
|
state.localState = cssMode.startState(htmlMode.indent(state.htmlState, ""));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
function maybeBackup(stream, pat, style) {
|
||||||
|
var cur = stream.current();
|
||||||
|
var close = cur.search(pat), m;
|
||||||
|
if (close > -1) stream.backUp(cur.length - close);
|
||||||
|
else if (m = cur.match(/<\/?$/)) {
|
||||||
|
stream.backUp(cur.length);
|
||||||
|
if (!stream.match(pat, false)) stream.match(cur[0]);
|
||||||
|
}
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
function javascript(stream, state) {
|
||||||
|
if (stream.match(/^<\/\s*script\s*>/i, false)) {
|
||||||
|
state.token = html;
|
||||||
|
state.localState = null;
|
||||||
|
return html(stream, state);
|
||||||
|
}
|
||||||
|
return maybeBackup(stream, /<\/\s*script\s*>/,
|
||||||
|
jsMode.token(stream, state.localState));
|
||||||
|
}
|
||||||
|
function css(stream, state) {
|
||||||
|
if (stream.match(/^<\/\s*style\s*>/i, false)) {
|
||||||
|
state.token = html;
|
||||||
|
state.localState = null;
|
||||||
|
return html(stream, state);
|
||||||
|
}
|
||||||
|
return maybeBackup(stream, /<\/\s*style\s*>/,
|
||||||
|
cssMode.token(stream, state.localState));
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState: function() {
|
||||||
|
var state = htmlMode.startState();
|
||||||
|
return {token: html, localState: null, mode: "html", htmlState: state};
|
||||||
|
},
|
||||||
|
|
||||||
|
copyState: function(state) {
|
||||||
|
if (state.localState)
|
||||||
|
var local = CodeMirror.copyState(state.token == css ? cssMode : jsMode, state.localState);
|
||||||
|
return {token: state.token, localState: local, mode: state.mode,
|
||||||
|
htmlState: CodeMirror.copyState(htmlMode, state.htmlState)};
|
||||||
|
},
|
||||||
|
|
||||||
|
token: function(stream, state) {
|
||||||
|
return state.token(stream, state);
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
if (state.token == html || /^\s*<\//.test(textAfter))
|
||||||
|
return htmlMode.indent(state.htmlState, textAfter);
|
||||||
|
else if (state.token == javascript)
|
||||||
|
return jsMode.indent(state.localState, textAfter);
|
||||||
|
else
|
||||||
|
return cssMode.indent(state.localState, textAfter);
|
||||||
|
},
|
||||||
|
|
||||||
|
electricChars: "/{}:",
|
||||||
|
|
||||||
|
innerMode: function(state) {
|
||||||
|
var mode = state.token == html ? htmlMode : state.token == javascript ? jsMode : cssMode;
|
||||||
|
return {state: state.localState || state.htmlState, mode: mode};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, "xml", "javascript", "css");
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/html", "htmlmixed");
|
52
codemirror-2.36/mode/htmlmixed/index.html
Normal file
52
codemirror-2.36/mode/htmlmixed/index.html
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: HTML mixed mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="../xml/xml.js"></script>
|
||||||
|
<script src="../javascript/javascript.js"></script>
|
||||||
|
<script src="../css/css.js"></script>
|
||||||
|
<script src="htmlmixed.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style>.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: HTML mixed mode</h1>
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<html style="color: green">
|
||||||
|
<!-- this is a comment -->
|
||||||
|
<head>
|
||||||
|
<title>Mixed HTML Example</title>
|
||||||
|
<style type="text/css">
|
||||||
|
h1 {font-family: comic sans; color: #f0f;}
|
||||||
|
div {background: yellow !important;}
|
||||||
|
body {
|
||||||
|
max-width: 50em;
|
||||||
|
margin: 1em 2em 1em 5em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Mixed HTML Example</h1>
|
||||||
|
<script>
|
||||||
|
function jsFunc(arg1, arg2) {
|
||||||
|
if (arg1 && arg2) document.body.innerHTML = "achoo";
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
</textarea></form>
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {mode: "text/html", tabMode: "indent"});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>The HTML mixed mode depends on the XML, JavaScript, and CSS modes.</p>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/html</code>
|
||||||
|
(redefined, only takes effect if you load this parser after the
|
||||||
|
XML parser).</p>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
87
codemirror-2.36/mode/javascript/index.html
Normal file
87
codemirror-2.36/mode/javascript/index.html
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: JavaScript mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="../../lib/util/continuecomment.js"></script>
|
||||||
|
<script src="javascript.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: JavaScript mode</h1>
|
||||||
|
|
||||||
|
<div><textarea id="code" name="code">
|
||||||
|
// Demo code (the actual new parser character stream implementation)
|
||||||
|
|
||||||
|
function StringStream(string) {
|
||||||
|
this.pos = 0;
|
||||||
|
this.string = string;
|
||||||
|
}
|
||||||
|
|
||||||
|
StringStream.prototype = {
|
||||||
|
done: function() {return this.pos >= this.string.length;},
|
||||||
|
peek: function() {return this.string.charAt(this.pos);},
|
||||||
|
next: function() {
|
||||||
|
if (this.pos < this.string.length)
|
||||||
|
return this.string.charAt(this.pos++);
|
||||||
|
},
|
||||||
|
eat: function(match) {
|
||||||
|
var ch = this.string.charAt(this.pos);
|
||||||
|
if (typeof match == "string") var ok = ch == match;
|
||||||
|
else var ok = ch && match.test ? match.test(ch) : match(ch);
|
||||||
|
if (ok) {this.pos++; return ch;}
|
||||||
|
},
|
||||||
|
eatWhile: function(match) {
|
||||||
|
var start = this.pos;
|
||||||
|
while (this.eat(match));
|
||||||
|
if (this.pos > start) return this.string.slice(start, this.pos);
|
||||||
|
},
|
||||||
|
backUp: function(n) {this.pos -= n;},
|
||||||
|
column: function() {return this.pos;},
|
||||||
|
eatSpace: function() {
|
||||||
|
var start = this.pos;
|
||||||
|
while (/\s/.test(this.string.charAt(this.pos))) this.pos++;
|
||||||
|
return this.pos - start;
|
||||||
|
},
|
||||||
|
match: function(pattern, consume, caseInsensitive) {
|
||||||
|
if (typeof pattern == "string") {
|
||||||
|
function cased(str) {return caseInsensitive ? str.toLowerCase() : str;}
|
||||||
|
if (cased(this.string).indexOf(cased(pattern), this.pos) == this.pos) {
|
||||||
|
if (consume !== false) this.pos += str.length;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var match = this.string.slice(this.pos).match(pattern);
|
||||||
|
if (match && consume !== false) this.pos += match[0].length;
|
||||||
|
return match;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</textarea></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
extraKeys: {"Enter": "newlineAndIndentContinueComment"}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
JavaScript mode supports a two configuration
|
||||||
|
options:
|
||||||
|
<ul>
|
||||||
|
<li><code>json</code> which will set the mode to expect JSON data rather than a JavaScript program.</li>
|
||||||
|
<li>
|
||||||
|
<code>typescript</code> which will activate additional syntax highlighting and some other things for TypeScript code (<a href="typescript.html">demo</a>).
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/javascript</code>, <code>application/json</code>, <code>text/typescript</code>, <code>application/typescript</code>.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
411
codemirror-2.36/mode/javascript/javascript.js
Normal file
411
codemirror-2.36/mode/javascript/javascript.js
Normal file
|
@ -0,0 +1,411 @@
|
||||||
|
// TODO actually recognize syntax of TypeScript constructs
|
||||||
|
|
||||||
|
CodeMirror.defineMode("javascript", function(config, parserConfig) {
|
||||||
|
var indentUnit = config.indentUnit;
|
||||||
|
var jsonMode = parserConfig.json;
|
||||||
|
var isTS = parserConfig.typescript;
|
||||||
|
|
||||||
|
// Tokenizer
|
||||||
|
|
||||||
|
var keywords = function(){
|
||||||
|
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"};
|
||||||
|
|
||||||
|
var jsKeywords = {
|
||||||
|
"if": A, "while": A, "with": A, "else": B, "do": B, "try": B, "finally": B,
|
||||||
|
"return": C, "break": C, "continue": C, "new": C, "delete": C, "throw": C,
|
||||||
|
"var": kw("var"), "const": kw("var"), "let": kw("var"),
|
||||||
|
"function": kw("function"), "catch": kw("catch"),
|
||||||
|
"for": kw("for"), "switch": kw("switch"), "case": kw("case"), "default": kw("default"),
|
||||||
|
"in": operator, "typeof": operator, "instanceof": operator,
|
||||||
|
"true": atom, "false": atom, "null": atom, "undefined": atom, "NaN": atom, "Infinity": atom
|
||||||
|
};
|
||||||
|
|
||||||
|
// Extend the 'normal' keywords with the TypeScript language extensions
|
||||||
|
if (isTS) {
|
||||||
|
var type = {type: "variable", style: "variable-3"};
|
||||||
|
var tsKeywords = {
|
||||||
|
// object-like things
|
||||||
|
"interface": kw("interface"),
|
||||||
|
"class": kw("class"),
|
||||||
|
"extends": kw("extends"),
|
||||||
|
"constructor": kw("constructor"),
|
||||||
|
|
||||||
|
// scope modifiers
|
||||||
|
"public": kw("public"),
|
||||||
|
"private": kw("private"),
|
||||||
|
"protected": kw("protected"),
|
||||||
|
"static": kw("static"),
|
||||||
|
|
||||||
|
"super": kw("super"),
|
||||||
|
|
||||||
|
// types
|
||||||
|
"string": type, "number": type, "bool": type, "any": type
|
||||||
|
};
|
||||||
|
|
||||||
|
for (var attr in tsKeywords) {
|
||||||
|
jsKeywords[attr] = tsKeywords[attr];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return jsKeywords;
|
||||||
|
}();
|
||||||
|
|
||||||
|
var isOperatorChar = /[+\-*&%=<>!?|]/;
|
||||||
|
|
||||||
|
function chain(stream, state, f) {
|
||||||
|
state.tokenize = f;
|
||||||
|
return f(stream, state);
|
||||||
|
}
|
||||||
|
|
||||||
|
function nextUntilUnescaped(stream, end) {
|
||||||
|
var escaped = false, next;
|
||||||
|
while ((next = stream.next()) != null) {
|
||||||
|
if (next == end && !escaped)
|
||||||
|
return false;
|
||||||
|
escaped = !escaped && next == "\\";
|
||||||
|
}
|
||||||
|
return escaped;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 jsTokenBase(stream, state) {
|
||||||
|
var ch = stream.next();
|
||||||
|
if (ch == '"' || ch == "'")
|
||||||
|
return chain(stream, state, jsTokenString(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 (ch == "/") {
|
||||||
|
if (stream.eat("*")) {
|
||||||
|
return chain(stream, state, jsTokenComment);
|
||||||
|
}
|
||||||
|
else if (stream.eat("/")) {
|
||||||
|
stream.skipToEnd();
|
||||||
|
return ret("comment", "comment");
|
||||||
|
}
|
||||||
|
else if (state.lastType == "operator" || state.lastType == "keyword c" ||
|
||||||
|
/^[\[{}\(,;:]$/.test(state.lastType)) {
|
||||||
|
nextUntilUnescaped(stream, "/");
|
||||||
|
stream.eatWhile(/[gimy]/); // 'y' is "sticky" option in Mozilla
|
||||||
|
return ret("regexp", "string-2");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
stream.eatWhile(isOperatorChar);
|
||||||
|
return ret("operator", null, stream.current());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (ch == "#") {
|
||||||
|
stream.skipToEnd();
|
||||||
|
return ret("error", "error");
|
||||||
|
}
|
||||||
|
else if (isOperatorChar.test(ch)) {
|
||||||
|
stream.eatWhile(isOperatorChar);
|
||||||
|
return ret("operator", null, stream.current());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
stream.eatWhile(/[\w\$_]/);
|
||||||
|
var word = stream.current(), known = keywords.propertyIsEnumerable(word) && keywords[word];
|
||||||
|
return (known && state.lastType != ".") ? ret(known.type, known.style, word) :
|
||||||
|
ret("variable", "variable", word);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function jsTokenString(quote) {
|
||||||
|
return function(stream, state) {
|
||||||
|
if (!nextUntilUnescaped(stream, quote))
|
||||||
|
state.tokenize = jsTokenBase;
|
||||||
|
return ret("string", "string");
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function jsTokenComment(stream, state) {
|
||||||
|
var maybeEnd = false, ch;
|
||||||
|
while (ch = stream.next()) {
|
||||||
|
if (ch == "/" && maybeEnd) {
|
||||||
|
state.tokenize = jsTokenBase;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
maybeEnd = (ch == "*");
|
||||||
|
}
|
||||||
|
return ret("comment", "comment");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parser
|
||||||
|
|
||||||
|
var atomicTypes = {"atom": true, "number": true, "variable": true, "string": true, "regexp": true};
|
||||||
|
|
||||||
|
function JSLexical(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 parseJS(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() : jsonMode ? expression : 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";
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 register(varname) {
|
||||||
|
var state = cx.state;
|
||||||
|
if (state.context) {
|
||||||
|
cx.marked = "def";
|
||||||
|
for (var v = state.localVars; v; v = v.next)
|
||||||
|
if (v.name == varname) return;
|
||||||
|
state.localVars = {name: varname, next: state.localVars};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Combinators
|
||||||
|
|
||||||
|
var defaultVars = {name: "this", next: {name: "arguments"}};
|
||||||
|
function pushcontext() {
|
||||||
|
cx.state.context = {prev: cx.state.context, vars: cx.state.localVars};
|
||||||
|
cx.state.localVars = defaultVars;
|
||||||
|
}
|
||||||
|
function popcontext() {
|
||||||
|
cx.state.localVars = cx.state.context.vars;
|
||||||
|
cx.state.context = cx.state.context.prev;
|
||||||
|
}
|
||||||
|
function pushlex(type, info) {
|
||||||
|
var result = function() {
|
||||||
|
var state = cx.state;
|
||||||
|
state.lexical = new JSLexical(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) {
|
||||||
|
return function expecting(type) {
|
||||||
|
if (type == wanted) return cont();
|
||||||
|
else if (wanted == ";") return pass();
|
||||||
|
else return cont(arguments.callee);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function statement(type) {
|
||||||
|
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("}"), block, poplex);
|
||||||
|
if (type == ";") return cont();
|
||||||
|
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);
|
||||||
|
return pass(pushlex("stat"), expression, expect(";"), poplex);
|
||||||
|
}
|
||||||
|
function expression(type) {
|
||||||
|
if (atomicTypes.hasOwnProperty(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(expression, "]"), 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" && value == "?") return cont(expression, expect(":"), 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 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 commaSeparated(type) {
|
||||||
|
if (type == end) return cont();
|
||||||
|
else return pass(what, proceed);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
function block(type) {
|
||||||
|
if (type == "}") return cont();
|
||||||
|
return pass(statement, block);
|
||||||
|
}
|
||||||
|
function maybetype(type) {
|
||||||
|
if (type == ":") return cont(typedef);
|
||||||
|
return pass();
|
||||||
|
}
|
||||||
|
function typedef(type) {
|
||||||
|
if (type == "variable"){cx.marked = "variable-3"; return cont();}
|
||||||
|
return pass();
|
||||||
|
}
|
||||||
|
function vardef1(type, value) {
|
||||||
|
if (type == "variable") {
|
||||||
|
register(value);
|
||||||
|
return isTS ? cont(maybetype, vardef2) : cont(vardef2);
|
||||||
|
}
|
||||||
|
return pass();
|
||||||
|
}
|
||||||
|
function vardef2(type, value) {
|
||||||
|
if (value == "=") return cont(expression, vardef2);
|
||||||
|
if (type == ",") return cont(vardef1);
|
||||||
|
}
|
||||||
|
function forspec1(type) {
|
||||||
|
if (type == "var") return cont(vardef1, expect(";"), forspec2);
|
||||||
|
if (type == ";") return cont(forspec2);
|
||||||
|
if (type == "variable") return cont(formaybein);
|
||||||
|
return cont(forspec2);
|
||||||
|
}
|
||||||
|
function formaybein(type, value) {
|
||||||
|
if (value == "in") return cont(expression);
|
||||||
|
return cont(maybeoperator, forspec2);
|
||||||
|
}
|
||||||
|
function forspec2(type, value) {
|
||||||
|
if (type == ";") return cont(forspec3);
|
||||||
|
if (value == "in") return cont(expression);
|
||||||
|
return cont(expression, expect(";"), forspec3);
|
||||||
|
}
|
||||||
|
function forspec3(type) {
|
||||||
|
if (type != ")") cont(expression);
|
||||||
|
}
|
||||||
|
function functiondef(type, value) {
|
||||||
|
if (type == "variable") {register(value); return cont(functiondef);}
|
||||||
|
if (type == "(") return cont(pushlex(")"), pushcontext, commasep(funarg, ")"), poplex, statement, popcontext);
|
||||||
|
}
|
||||||
|
function funarg(type, value) {
|
||||||
|
if (type == "variable") {register(value); return isTS ? cont(maybetype) : cont();}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface
|
||||||
|
|
||||||
|
return {
|
||||||
|
startState: function(basecolumn) {
|
||||||
|
return {
|
||||||
|
tokenize: jsTokenBase,
|
||||||
|
lastType: null,
|
||||||
|
cc: [],
|
||||||
|
lexical: new JSLexical((basecolumn || 0) - indentUnit, 0, "block", false),
|
||||||
|
localVars: parserConfig.localVars,
|
||||||
|
context: parserConfig.localVars && {vars: parserConfig.localVars},
|
||||||
|
indented: 0
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
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.lastType = type;
|
||||||
|
return parseJS(state, style, type, content, stream);
|
||||||
|
},
|
||||||
|
|
||||||
|
indent: function(state, textAfter) {
|
||||||
|
if (state.tokenize == jsTokenComment) return CodeMirror.Pass;
|
||||||
|
if (state.tokenize != jsTokenBase) 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 + (state.lastType == "operator" || state.lastType == "," ? 4 : 0);
|
||||||
|
else if (type == "form" && firstChar == "{") return lexical.indented;
|
||||||
|
else if (type == "form") return lexical.indented + indentUnit;
|
||||||
|
else if (type == "stat")
|
||||||
|
return lexical.indented + (state.lastType == "operator" || state.lastType == "," ? indentUnit : 0);
|
||||||
|
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: ":{}",
|
||||||
|
|
||||||
|
jsonMode: jsonMode
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
CodeMirror.defineMIME("text/javascript", "javascript");
|
||||||
|
CodeMirror.defineMIME("application/json", {name: "javascript", json: true});
|
||||||
|
CodeMirror.defineMIME("text/typescript", { name: "javascript", typescript: true });
|
||||||
|
CodeMirror.defineMIME("application/typescript", { name: "javascript", typescript: true });
|
48
codemirror-2.36/mode/javascript/typescript.html
Normal file
48
codemirror-2.36/mode/javascript/typescript.html
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: TypeScript mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="javascript.js"></script>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: TypeScript mode</h1>
|
||||||
|
|
||||||
|
<div><textarea id="code" name="code">
|
||||||
|
class Greeter {
|
||||||
|
greeting: string;
|
||||||
|
constructor (message: string) {
|
||||||
|
this.greeting = message;
|
||||||
|
}
|
||||||
|
greet() {
|
||||||
|
return "Hello, " + this.greeting;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var greeter = new Greeter("world");
|
||||||
|
|
||||||
|
var button = document.createElement('button')
|
||||||
|
button.innerText = "Say Hello"
|
||||||
|
button.onclick = function() {
|
||||||
|
alert(greeter.greet())
|
||||||
|
}
|
||||||
|
|
||||||
|
document.body.appendChild(button)
|
||||||
|
|
||||||
|
</textarea></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
lineNumbers: true,
|
||||||
|
matchBrackets: true,
|
||||||
|
mode: "text/typescript"
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>This is a specialization of the <a href="index.html">JavaScript mode</a>.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
38
codemirror-2.36/mode/jinja2/index.html
Normal file
38
codemirror-2.36/mode/jinja2/index.html
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: Jinja2 mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="jinja2.js"></script>
|
||||||
|
<style type="text/css">.CodeMirror {border-top: 1px solid black; border-bottom: 1px solid black;}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: Jinja2 mode</h1>
|
||||||
|
<form><textarea id="code" name="code">
|
||||||
|
<html style="color: green">
|
||||||
|
<!-- this is a comment -->
|
||||||
|
<head>
|
||||||
|
<title>Jinja2 Example</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<ul>
|
||||||
|
{# this is a comment #}
|
||||||
|
{%- for item in li -%}
|
||||||
|
<li>
|
||||||
|
{{ item.label }}
|
||||||
|
</li>
|
||||||
|
{% endfor -%}
|
||||||
|
</ul>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
</textarea></form>
|
||||||
|
<script>
|
||||||
|
var editor =
|
||||||
|
CodeMirror.fromTextArea(document.getElementById("code"), {mode:
|
||||||
|
{name: "jinja2", htmlMode: true}});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
42
codemirror-2.36/mode/jinja2/jinja2.js
Normal file
42
codemirror-2.36/mode/jinja2/jinja2.js
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
CodeMirror.defineMode("jinja2", function(config, parserConf) {
|
||||||
|
var keywords = ["block", "endblock", "for", "endfor", "in", "true", "false",
|
||||||
|
"loop", "none", "self", "super", "if", "as", "not", "and",
|
||||||
|
"else", "import", "with", "without", "context"];
|
||||||
|
keywords = new RegExp("^((" + keywords.join(")|(") + "))\\b");
|
||||||
|
|
||||||
|
function tokenBase (stream, state) {
|
||||||
|
var ch = stream.next();
|
||||||
|
if (ch == "{") {
|
||||||
|
if (ch = stream.eat(/\{|%|#/)) {
|
||||||
|
stream.eat("-");
|
||||||
|
state.tokenize = inTag(ch);
|
||||||
|
return "tag";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function inTag (close) {
|
||||||
|
if (close == "{") {
|
||||||
|
close = "}";
|
||||||
|
}
|
||||||
|
return function (stream, state) {
|
||||||
|
var ch = stream.next();
|
||||||
|
if ((ch == close || (ch == "-" && stream.eat(close)))
|
||||||
|
&& stream.eat("}")) {
|
||||||
|
state.tokenize = tokenBase;
|
||||||
|
return "tag";
|
||||||
|
}
|
||||||
|
if (stream.match(keywords)) {
|
||||||
|
return "keyword";
|
||||||
|
}
|
||||||
|
return close == "#" ? "comment" : "string";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
startState: function () {
|
||||||
|
return {tokenize: tokenBase};
|
||||||
|
},
|
||||||
|
token: function (stream, state) {
|
||||||
|
return state.tokenize(stream, state);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
740
codemirror-2.36/mode/less/index.html
Normal file
740
codemirror-2.36/mode/less/index.html
Normal file
|
@ -0,0 +1,740 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>CodeMirror: LESS mode</title>
|
||||||
|
<link rel="stylesheet" href="../../lib/codemirror.css">
|
||||||
|
<script src="../../lib/codemirror.js"></script>
|
||||||
|
<script src="less.js"></script>
|
||||||
|
<style>.CodeMirror {background: #f8f8f8; border: 1px solid #ddd; font-size:12px} .CodeMirror-scroll {height: 400px}</style>
|
||||||
|
<link rel="stylesheet" href="../../doc/docs.css">
|
||||||
|
<link rel="stylesheet" href="../../theme/lesser-dark.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>CodeMirror: LESS mode</h1>
|
||||||
|
<form><textarea id="code" name="code">@media screen and (device-aspect-ratio: 16/9) { … }
|
||||||
|
@media screen and (device-aspect-ratio: 32/18) { … }
|
||||||
|
@media screen and (device-aspect-ratio: 1280/720) { … }
|
||||||
|
@media screen and (device-aspect-ratio: 2560/1440) { … }
|
||||||
|
|
||||||
|
html:lang(fr-be)
|
||||||
|
html:lang(de)
|
||||||
|
:lang(fr-be) > q
|
||||||
|
:lang(de) > q
|
||||||
|
|
||||||
|
tr:nth-child(2n+1) /* represents every odd row of an HTML table */
|
||||||
|
tr:nth-child(odd) /* same */
|
||||||
|
tr:nth-child(2n+0) /* represents every even row of an HTML table */
|
||||||
|
tr:nth-child(even) /* same */
|
||||||
|
|
||||||
|
/* Alternate paragraph colours in CSS */
|
||||||
|
p:nth-child(4n+1) { color: navy; }
|
||||||
|
p:nth-child(4n+2) { color: green; }
|
||||||
|
p:nth-child(4n+3) { color: maroon; }
|
||||||
|
p:nth-child(4n+4) { color: purple; }
|
||||||
|
|
||||||
|
:nth-child(10n-1) /* represents the 9th, 19th, 29th, etc, element */
|
||||||
|
:nth-child(10n+9) /* Same */
|
||||||
|
:nth-child(10n+-1) /* Syntactically invalid, and would be ignored */
|
||||||
|
|
||||||
|
:nth-child( 3n + 1 )
|
||||||
|
:nth-child( +3n - 2 )
|
||||||
|
:nth-child( -n+ 6)
|
||||||
|
:nth-child( +6 )
|
||||||
|
|
||||||
|
html|tr:nth-child(-n+6) /* represents the 6 first rows of XHTML tables */
|
||||||
|
|
||||||
|
img:nth-of-type(2n+1) { float: right; }
|
||||||
|
img:nth-of-type(2n) { float: left; }
|
||||||
|
|
||||||
|
body > h2:nth-of-type(n+2):nth-last-of-type(n+2)
|
||||||
|
body > h2:not(:first-of-type):not(:last-of-type)
|
||||||
|
|
||||||
|
html|*:not(:link):not(:visited)
|
||||||
|
*|*:not(:hover)
|
||||||
|
p::first-line { text-transform: uppercase }
|
||||||
|
|
||||||
|
p { color: red; font-size: 12pt }
|
||||||
|
p::first-letter { color: green; font-size: 200% }
|
||||||
|
p::first-line { color: blue }
|
||||||
|
|
||||||
|
p { line-height: 1.1 }
|
||||||
|
p::first-letter { font-size: 3em; font-weight: normal }
|
||||||
|
span { font-weight: bold }
|
||||||
|
|
||||||
|
* /* a=0 b=0 c=0 -> specificity = 0 */
|
||||||
|
LI /* a=0 b=0 c=1 -> specificity = 1 */
|
||||||
|
UL LI /* a=0 b=0 c=2 -> specificity = 2 */
|
||||||
|
UL OL+LI /* a=0 b=0 c=3 -> specificity = 3 */
|
||||||
|
H1 + *[REL=up] /* a=0 b=1 c=1 -> specificity = 11 */
|
||||||
|
UL OL LI.red /* a=0 b=1 c=3 -> specificity = 13 */
|
||||||
|
LI.red.level /* a=0 b=2 c=1 -> specificity = 21 */
|
||||||
|
#x34y /* a=1 b=0 c=0 -> specificity = 100 */
|
||||||
|
#s12:not(FOO) /* a=1 b=0 c=1 -> specificity = 101 */
|
||||||
|
|
||||||
|
@namespace foo url(http://www.example.com);
|
||||||
|
foo|h1 { color: blue } /* first rule */
|
||||||
|
foo|* { color: yellow } /* second rule */
|
||||||
|
|h1 { color: red } /* ...*/
|
||||||
|
*|h1 { color: green }
|
||||||
|
h1 { color: green }
|
||||||
|
|
||||||
|
span[hello="Ocean"][goodbye="Land"]
|
||||||
|
|
||||||
|
a[rel~="copyright"] { ... }
|
||||||
|
a[href="http://www.w3.org/"] { ... }
|
||||||
|
|
||||||
|
DIALOGUE[character=romeo]
|
||||||
|
DIALOGUE[character=juliet]
|
||||||
|
|
||||||
|
[att^=val]
|
||||||
|
[att$=val]
|
||||||
|
[att*=val]
|
||||||
|
|
||||||
|
@namespace foo "http://www.example.com";
|
||||||
|
[foo|att=val] { color: blue }
|
||||||
|
[*|att] { color: yellow }
|
||||||
|
[|att] { color: green }
|
||||||
|
[att] { color: green }
|
||||||
|
|
||||||
|
|
||||||
|
*:target { color : red }
|
||||||
|
*:target::before { content : url(target.png) }
|
||||||
|
|
||||||
|
E[foo]{
|
||||||
|
padding:65px;
|
||||||
|
}
|
||||||
|
E[foo] ~ F{
|
||||||
|
padding:65px;
|
||||||
|
}
|
||||||
|
E#myid{
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
.el tr:nth-child(even):last-child td:first-child{
|
||||||
|
-moz-border-radius-bottomleft:3px;
|
||||||
|
-webkit-border-bottom-left-radius:3px;
|
||||||
|
border-bottom-left-radius:3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Some LESS code */
|
||||||
|
|
||||||
|
button {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
border: 0;
|
||||||
|
margin: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
button.icon-plus { background: url(http://dahlström.net/tmp/sharp-icons/svg-icon-target.svg#plus) no-repeat; }
|
||||||
|
button.icon-chart { background: url(http://dahlström.net/tmp/sharp-icons/svg-icon-target.svg#chart) no-repeat; }
|
||||||
|
|
||||||
|
button:hover { background-color: #999; }
|
||||||
|
button:active { background-color: #666; }
|
||||||
|
|
||||||
|
@test_a: #eeeQQQ;//this is not a valid hex value and thus parsed as an element id
|
||||||
|
@test_b: #eeeFFF //this is a valid hex value but the declaration doesn't end with a semicolon and thus parsed as an element id
|
||||||
|
|
||||||
|
#eee aaa .box
|
||||||
|
{
|
||||||
|
#test bbb {
|
||||||
|
width: 500px;
|
||||||
|
height: 250px;
|
||||||
|
background-image: url(dir/output/sheep.png), url( betweengrassandsky.png );
|
||||||
|
background-position: center bottom, left top;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@base: #f938ab;
|
||||||
|
|
||||||
|
.box-shadow(@style, @c) when (iscolor(@c)) {
|
||||||
|
box-shadow: @style @c;
|
||||||
|
-webkit-box-shadow: @style @c;
|
||||||
|
-moz-box-shadow: @style @c;
|
||||||
|
}
|
||||||
|
.box-shadow(@style, @alpha: 50%) when (isnumber(@alpha)) {
|
||||||
|
.box-shadow(@style, rgba(0, 0, 0, @alpha));
|
||||||
|
}
|
||||||
|
|
||||||
|
@color: #4D926F;
|
||||||
|
|
||||||
|
#header {
|
||||||
|
color: @color;
|
||||||
|
color: #000000;
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
color: @color;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rounded-corners (@radius: 5px) {
|
||||||
|
border-radius: @radius;
|
||||||
|
-webkit-border-radius: @radius;
|
||||||
|
-moz-border-radius: @radius;
|
||||||
|
}
|
||||||
|
|
||||||
|
#header {
|
||||||
|
.rounded-corners;
|
||||||
|
}
|
||||||
|
#footer {
|
||||||
|
.rounded-corners(10px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-shadow (@x: 0, @y: 0, @blur: 1px, @alpha) {
|
||||||
|
@val: @x @y @blur rgba(0, 0, 0, @alpha);
|
||||||
|
|
||||||
|
box-shadow: @val;
|
||||||
|
-webkit-box-shadow: @val;
|
||||||
|
-moz-box-shadow: @val;
|
||||||
|
}
|
||||||
|
.box { @base: #f938ab;
|
||||||
|
color: saturate(@base, 5%);
|
||||||
|
border-color: lighten(@base, 30%);
|
||||||
|
div { .box-shadow(0, 0, 5px, 0.4) }
|
||||||
|
}
|
||||||
|
|
||||||
|
@import url("something.css");
|
||||||
|
|
||||||
|
@light-blue: hsl(190, 50%, 65%);
|
||||||
|
@light-yellow: desaturate(#fefec8, 10%);
|
||||||
|
@dark-yellow: desaturate(darken(@light-yellow, 10%), 40%);
|
||||||
|
@darkest: hsl(20, 0%, 15%);
|
||||||
|
@dark: hsl(190, 20%, 30%);
|
||||||
|
@medium: hsl(10, 60%, 30%);
|
||||||
|
@light: hsl(90, 40%, 20%);
|
||||||
|
@lightest: hsl(90, 20%, 90%);
|
||||||
|
@highlight: hsl(80, 50%, 90%);
|
||||||
|
@blue: hsl(210, 60%, 20%);
|
||||||
|
@alpha-blue: hsla(210, 60%, 40%, 0.5);
|
||||||
|
|
||||||
|
.box-shadow (@x, @y, @blur, @alpha) {
|
||||||
|
@value: @x @y @blur rgba(0, 0, 0, @alpha);
|
||||||
|
box-shadow: @value;
|
||||||
|
-moz-box-shadow: @value;
|
||||||
|
-webkit-box-shadow: @value;
|
||||||
|
}
|
||||||
|
.border-radius (@radius) {
|
||||||
|
border-radius: @radius;
|
||||||
|
-moz-border-radius: @radius;
|
||||||
|
-webkit-border-radius: @radius;
|
||||||
|
}
|
||||||
|
|
||||||
|
.border-radius (@radius, bottom) {
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
-moz-border-top-right-radius: 0;
|
||||||
|
-moz-border-top-left-radius: 0;
|
||||||
|
-webkit-border-top-left-radius: 0;
|
||||||
|
-webkit-border-top-right-radius: 0;
|
||||||
|
}
|
||||||
|
.border-radius (@radius, right) {
|
||||||
|
border-bottom-left-radius: 0;
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
-moz-border-bottom-left-radius: 0;
|
||||||
|
-moz-border-top-left-radius: 0;
|
||||||
|
-webkit-border-bottom-left-radius: 0;
|
||||||
|
-webkit-border-top-left-radius: 0;
|
||||||
|
}
|
||||||
|
.box-shadow-inset (@x, @y, @blur, @color) {
|
||||||
|
box-shadow: @x @y @blur @color inset;
|
||||||
|
-moz-box-shadow: @x @y @blur @color inset;
|
||||||
|
-webkit-box-shadow: @x @y @blur @color inset;
|
||||||
|
}
|
||||||
|
.code () {
|
||||||
|
font-family: 'Bitstream Vera Sans Mono',
|
||||||
|
'DejaVu Sans Mono',
|
||||||
|
'Monaco',
|
||||||
|
Courier,
|
||||||
|
monospace !important;
|
||||||
|
}
|
||||||
|
.wrap () {
|
||||||
|
text-wrap: wrap;
|
||||||
|
white-space: pre-wrap; /* css-3 */
|
||||||
|
white-space: -moz-pre-wrap; /* Mozilla, since 1999 */
|
||||||
|
white-space: -pre-wrap; /* Opera 4-6 */
|
||||||
|
white-space: -o-pre-wrap; /* Opera 7 */
|
||||||
|
word-wrap: break-word; /* Internet Explorer 5.5+ */
|
||||||
|
}
|
||||||
|
|
||||||
|
html { margin: 0 }
|
||||||
|
body {
|
||||||
|
background-color: @darkest;
|
||||||
|
margin: 0 auto;
|
||||||
|
font-family: Arial, sans-serif;
|
||||||
|
font-size: 100%;
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
nav, header, footer, section, article {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
color: #b83000;
|
||||||
|
}
|
||||||
|
h1 a {
|
||||||
|
color: black;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
a:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
h1, h2, h3, h4 {
|
||||||
|
margin: 0;
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
ul, li {
|
||||||
|
list-style-type: none;
|
||||||
|
}
|
||||||
|
code { .code; }
|
||||||
|
code {
|
||||||
|
.string, .regexp { color: @dark }
|
||||||
|
.keyword { font-weight: bold }
|
||||||
|
.comment { color: rgba(0, 0, 0, 0.5) }
|
||||||
|
.number { color: @blue }
|
||||||
|
.class, .special { color: rgba(0, 50, 100, 0.8) }
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
padding: 0 30px;
|
||||||
|
.wrap;
|
||||||
|
}
|
||||||
|
blockquote {
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
body > footer {
|
||||||
|
text-align: left;
|
||||||
|
margin-left: 10px;
|
||||||
|
font-style: italic;
|
||||||
|
font-size: 18px;
|
||||||
|
color: #888;
|
||||||
|
}
|
||||||
|
|
||||||
|
#logo {
|
||||||
|
margin-top: 30px;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
display: block;
|
||||||
|
width: 199px;
|
||||||
|
height: 81px;
|
||||||
|
background: url(/images/logo.png) no-repeat;
|
||||||
|
}
|
||||||
|
nav {
|
||||||
|
margin-left: 15px;
|
||||||
|
}
|
||||||
|
nav a, #dropdown li {
|
||||||
|
display: inline-block;
|
||||||
|
color: white;
|
||||||
|
line-height: 42px;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0px 15px;
|
||||||
|
text-shadow: -1px -1px 1px rgba(0, 0, 0, 0.5);
|
||||||
|
text-decoration: none;
|
||||||
|
border: 2px solid transparent;
|
||||||
|
border-width: 0 2px;
|
||||||
|
&:hover {
|
||||||
|
.dark-red;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.dark-red {
|
||||||
|
@red: @medium;
|
||||||
|
border: 2px solid darken(@red, 25%);
|
||||||
|
border-left-color: darken(@red, 15%);
|
||||||
|
border-right-color: darken(@red, 15%);
|
||||||
|
border-bottom: 0;
|
||||||
|
border-top: 0;
|
||||||
|
background-color: darken(@red, 10%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 980px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#main {
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 100%;
|
||||||
|
background-color: @light-blue;
|
||||||
|
border-top: 8px solid darken(@light-blue, 5%);
|
||||||
|
|
||||||
|
#intro {
|
||||||
|
background-color: lighten(@light-blue, 25%);
|
||||||
|
float: left;
|
||||||
|
margin-top: -8px;
|
||||||
|
margin-right: 5px;
|
||||||
|
|
||||||
|
height: 380px;
|
||||||
|
position: relative;
|
||||||
|
z-index: 2;
|
||||||
|
font-family: 'Droid Serif', 'Georgia';
|
||||||
|
width: 395px;
|
||||||
|
padding: 45px 20px 23px 30px;
|
||||||
|
border: 2px dashed darken(@light-blue, 10%);
|
||||||
|
.box-shadow(1px, 0px, 6px, 0.5);
|
||||||
|
border-bottom: 0;
|
||||||
|
border-top: 0;
|
||||||
|
#download { color: transparent; border: 0; float: left; display: inline-block; margin: 15px 0 15px -5px; }
|
||||||
|
#download img { display: inline-block}
|
||||||
|
#download-info {
|
||||||
|
code {
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
color: @blue + #333; display: inline; float: left; margin: 36px 0 0 15px }
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
span {
|
||||||
|
color: @medium;
|
||||||
|
}
|
||||||
|
color: @blue;
|
||||||
|
margin: 20px 0;
|
||||||
|
font-size: 24px;
|
||||||
|
line-height: 1.2em;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
color: @blue;
|
||||||
|
line-height: 1.4em;
|
||||||
|
margin: 30px 0 15px 0;
|
||||||
|
font-size: 1em;
|
||||||
|
text-shadow: 0px 0px 0px @lightest;
|
||||||
|
span { color: @medium }
|
||||||
|
}
|
||||||
|
#example {
|
||||||
|
p {
|
||||||
|
font-size: 18px;
|
||||||
|
color: @blue;
|
||||||
|
font-weight: bold;
|
||||||
|
text-shadow: 0px 1px 1px @lightest;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
margin: 0;
|
||||||
|
text-shadow: 0 -1px 1px @darkest;
|
||||||
|
margin-top: 20px;
|
||||||
|
background-color: desaturate(@darkest, 8%);
|
||||||
|
border: 0;
|
||||||
|
width: 450px;
|
||||||
|
color: lighten(@lightest, 2%);
|
||||||
|
background-repeat: repeat;
|
||||||
|
padding: 15px;
|
||||||
|
border: 1px dashed @lightest;
|
||||||
|
line-height: 15px;
|
||||||
|
.box-shadow(0, 0px, 15px, 0.5);
|
||||||
|
.code;
|
||||||
|
.border-radius(2px);
|
||||||
|
code .attribute { color: hsl(40, 50%, 70%) }
|
||||||
|
code .variable { color: hsl(120, 10%, 50%) }
|
||||||
|
code .element { color: hsl(170, 20%, 50%) }
|
||||||
|
|
||||||
|
code .string, .regexp { color: hsl(75, 50%, 65%) }
|
||||||
|
code .class { color: hsl(40, 40%, 60%); font-weight: normal }
|
||||||
|
code .id { color: hsl(50, 40%, 60%); font-weight: normal }
|
||||||
|
code .comment { color: rgba(255, 255, 255, 0.2) }
|
||||||
|
code .number, .color { color: hsl(10, 40%, 50%) }
|
||||||
|
code .class, code .mixin, .special { color: hsl(190, 20%, 50%) }
|
||||||
|
#time { color: #aaa }
|
||||||
|
}
|
||||||
|
float: right;
|
||||||
|
font-size: 12px;
|
||||||
|
margin: 0;
|
||||||
|
margin-top: 15px;
|
||||||
|
padding: 0;
|
||||||
|
width: 500px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.page {
|
||||||
|
.content {
|
||||||
|
width: 870px;
|
||||||
|
padding: 45px;
|
||||||
|
}
|
||||||
|
margin: 0 auto;
|
||||||
|
font-family: 'Georgia', serif;
|
||||||
|
font-size: 18px;
|
||||||
|
line-height: 26px;
|
||||||
|
padding: 0 60px;
|
||||||
|
code {
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
border-width: 1px;
|
||||||
|
border-style: dashed;
|
||||||
|
padding: 15px;
|
||||||
|
margin: 15px 0;
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
text-align: left;
|
||||||
|
font-size: 40px;
|
||||||
|
margin-top: 15px;
|
||||||
|
margin-bottom: 35px;
|
||||||
|
}
|
||||||
|
p + h1 { margin-top: 60px }
|
||||||
|
h2, h3 {
|
||||||
|
margin: 30px 0 15px 0;
|
||||||
|
}
|
||||||
|
p + h2, pre + h2, code + h2 {
|
||||||
|
border-top: 6px solid rgba(255, 255, 255, 0.1);
|
||||||
|
padding-top: 30px;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
margin: 15px 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#docs {
|
||||||
|
@bg: lighten(@light-blue, 5%);
|
||||||
|
border-top: 2px solid lighten(@bg, 5%);
|
||||||
|
color: @blue;
|
||||||
|
background-color: @light-blue;
|
||||||
|
.box-shadow(0, -2px, 5px, 0.2);
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-family: 'Droid Serif', 'Georgia', serif;
|
||||||
|
padding-top: 30px;
|
||||||
|
padding-left: 45px;
|
||||||
|
font-size: 44px;
|
||||||
|
text-align: left;
|
||||||
|
margin: 30px 0 !important;
|
||||||
|
text-shadow: 0px 1px 1px @lightest;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.content {
|
||||||
|
clear: both;
|
||||||
|
border-color: transparent;
|
||||||
|
background-color: lighten(@light-blue, 25%);
|
||||||
|
.box-shadow(0, 5px, 5px, 0.4);
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
@background: lighten(@bg, 30%);
|
||||||
|
color: lighten(@blue, 10%);
|
||||||
|
background-color: @background;
|
||||||
|
border-color: lighten(@light-blue, 25%);
|
||||||
|
border-width: 2px;
|
||||||
|
code .attribute { color: hsl(40, 50%, 30%) }
|
||||||
|
code .variable { color: hsl(120, 10%, 30%) }
|
||||||
|
code .element { color: hsl(170, 20%, 30%) }
|
||||||
|
|
||||||
|
code .string, .regexp { color: hsl(75, 50%, 35%) }
|
||||||
|
code .class { color: hsl(40, 40%, 30%); font-weight: normal }
|
||||||
|
code .id { color: hsl(50, 40%, 30%); font-weight: normal }
|
||||||
|
code .comment { color: rgba(0, 0, 0, 0.4) }
|
||||||
|
code .number, .color { color: hsl(10, 40%, 30%) }
|
||||||
|
code .class, code .mixin, .special { color: hsl(190, 20%, 30%) }
|
||||||
|
}
|
||||||
|
pre code { font-size: 15px }
|
||||||
|
p + h2, pre + h2, code + h2 { border-top-color: rgba(0, 0, 0, 0.1) }
|
||||||
|
}
|
||||||
|
|
||||||
|
td {
|
||||||
|
padding-right: 30px;
|
||||||
|
}
|
||||||
|
#synopsis {
|
||||||
|
.box-shadow(0, 5px, 5px, 0.2);
|
||||||
|
}
|
||||||
|
#synopsis, #about {
|
||||||
|
h2 {
|
||||||
|
font-size: 30px;
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
h1 + h2 {
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
h3 { font-size: 22px }
|
||||||
|
|
||||||
|
.code-example {
|
||||||
|
border-spacing: 0;
|
||||||
|
border-width: 1px;
|
||||||
|
border-style: dashed;
|
||||||
|
padding: 0;
|
||||||
|
pre { border: 0; margin: 0 }
|
||||||
|
td {
|
||||||
|
border: 0;
|
||||||
|
margin: 0;
|
||||||
|
background-color: desaturate(darken(@darkest, 5%), 20%);
|
||||||
|
vertical-align: top;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
tr { padding: 0 }
|
||||||
|
}
|
||||||
|
.css-output {
|
||||||
|
td {
|
||||||
|
border-left: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.less-example {
|
||||||
|
//border-right: 1px dotted rgba(255, 255, 255, 0.5) !important;
|
||||||
|
}
|
||||||
|
.css-output, .less-example {
|
||||||
|
width: 390px;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
padding: 20px;
|
||||||
|
line-height: 20px;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#about, #synopsis, #guide {
|
||||||
|
a {
|
||||||
|
text-decoration: none;
|
||||||
|
color: @light-yellow;
|
||||||
|
border-bottom: 1px dashed rgba(255, 255, 255, 0.2);
|
||||||
|
&:hover {
|
||||||
|
text-decoration: none;
|
||||||
|
border-bottom: 1px dashed @light-yellow;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@bg: desaturate(darken(@darkest, 5%), 20%);
|
||||||
|
text-shadow: 0 -1px 1px lighten(@bg, 5%);
|
||||||
|
color: @highlight;
|
||||||
|
background-color: @bg;
|
||||||
|
.content {
|
||||||
|
background-color: desaturate(@darkest, 20%);
|
||||||
|
clear: both;
|
||||||
|
.box-shadow(0, 5px, 5px, 0.4);
|
||||||
|
}
|
||||||
|
h1, h2, h3 {
|
||||||
|
color: @dark-yellow;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
code .attribute { color: hsl(40, 50%, 70%) }
|
||||||
|
code .variable { color: hsl(120, 10%, 50%) }
|
||||||
|
code .element { color: hsl(170, 20%, 50%) }
|
||||||
|
|
||||||
|
code .string, .regexp { color: hsl(75, 50%, 65%) }
|
||||||
|
code .class { color: hsl(40, 40%, 60%); font-weight: normal }
|
||||||
|
code .id { color: hsl(50, 40%, 60%); font-weight: normal }
|
||||||
|
code .comment { color: rgba(255, 255, 255, 0.2) }
|
||||||
|
code .number, .color { color: hsl(10, 40%, 50%) }
|
||||||
|
code .class, code .mixin, .special { color: hsl(190, 20%, 50%) }
|
||||||
|
background-color: @bg;
|
||||||
|
border-color: darken(@light-yellow, 5%);
|
||||||
|
}
|
||||||
|
code {
|
||||||
|
color: darken(@dark-yellow, 5%);
|
||||||
|
.string, .regexp { color: desaturate(@light-blue, 15%) }
|
||||||
|
.keyword { color: hsl(40, 40%, 60%); font-weight: normal }
|
||||||
|
.comment { color: rgba(255, 255, 255, 0.2) }
|
||||||
|
.number { color: lighten(@blue, 10%) }
|
||||||
|
.class, .special { color: hsl(190, 20%, 50%) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#guide {
|
||||||
|
background-color: @darkest;
|
||||||
|
.content {
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#about {
|
||||||
|
background-color: @darkest !important;
|
||||||
|
.content {
|
||||||
|
background-color: desaturate(lighten(@darkest, 3%), 5%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#synopsis {
|
||||||
|
background-color: desaturate(lighten(@darkest, 3%), 5%) !important;
|
||||||
|
.content {
|
||||||
|
background-color: desaturate(lighten(@darkest, 3%), 5%);
|
||||||
|
}
|
||||||
|
pre {}
|
||||||
|
}
|
||||||
|
#synopsis, #guide {
|
||||||
|
.content {
|
||||||
|
.box-shadow(0, 0px, 0px, 0.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#about footer {
|
||||||
|
margin-top: 30px;
|
||||||
|
padding-top: 30px;
|
||||||
|
border-top: 6px solid rgba(0, 0, 0, 0.1);
|
||||||
|
text-align: center;
|
||||||
|
font-size: 16px;
|
||||||
|
color: rgba(255, 255, 255, 0.35);
|
||||||
|
#copy { font-size: 12px }
|
||||||
|
text-shadow: -1px -1px 1px rgba(0, 0, 0, 0.02);
|
||||||
|
}
|
||||||
|
</textarea></form>
|
||||||
|
<script>
|
||||||
|
var editor = CodeMirror.fromTextArea(document.getElementById("code"), {
|
||||||
|
theme: "lesser-dark",
|
||||||
|
lineNumbers : true,
|
||||||
|
matchBrackets : true
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p><strong>MIME types defined:</strong> <code>text/x-less</code>, <code>text/css</code> (if not previously defined).</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user