2011-11-05 16:46:53 +00:00
|
|
|
'use strict';
|
|
|
|
|
2011-10-07 01:04:47 +00:00
|
|
|
/*@
|
|
|
|
Ox.doc <f> Generates documentation for annotated JavaScript
|
2012-04-09 08:39:02 +00:00
|
|
|
(source) -> <[o]> Array of doc objects
|
|
|
|
arguments <[o]|u> Arguments (array of doc objects)
|
|
|
|
Present if the <code>type</code> of the item is
|
|
|
|
<code>"function"</code>.
|
|
|
|
description <s|u> Multi-line description with optional markup
|
|
|
|
See Ox.parseHTML for details
|
|
|
|
events <[o]|u> Events (array of doc objects)
|
|
|
|
Present if the item fires any events
|
|
|
|
file <s> File name
|
|
|
|
line <n> Line number
|
|
|
|
name <s> Name of the item
|
|
|
|
properties <[o]|u> Properties (array of doc objects)
|
|
|
|
Present if the <code>type</code> of the item is
|
|
|
|
<code>"event"</code>, <code>"function"</code>
|
|
|
|
or <code>"object"</code>.
|
|
|
|
section <s|u> Section in the file
|
|
|
|
source <[o]> Source code (array of tokens)
|
|
|
|
length <n> Length of the token
|
|
|
|
offset <n> Offset of the token
|
|
|
|
type <s> Type of the token
|
|
|
|
See Ox.tokenize for list of types
|
|
|
|
summary <s> One-line summary
|
|
|
|
usage <[o]> Usage (array of doc objects)
|
|
|
|
Present if the <code>type</code> of the item is
|
|
|
|
<code>"function"</code>.
|
|
|
|
type <s> Type of the item
|
2011-10-07 01:04:47 +00:00
|
|
|
(file, callback) -> <u> undefined
|
2012-04-09 08:39:02 +00:00
|
|
|
(files, callback) -> <u> undefined
|
|
|
|
source <s> JavaScript source code
|
2011-10-07 01:04:47 +00:00
|
|
|
file <s> JavaScript file
|
2012-04-09 08:39:02 +00:00
|
|
|
files <[s]> Array of javascript files
|
2011-10-07 01:04:47 +00:00
|
|
|
callback <f> Callback function
|
|
|
|
doc <[o]> Array of doc objects
|
2012-04-09 08:39:02 +00:00
|
|
|
# > Ox.doc("//@ My.FOO <n> Magic constant\nMy.FOO = 23;")
|
|
|
|
# [{"name": "Ox.foo", "summary": "just some string", "type": "string"}]
|
2011-10-07 01:04:47 +00:00
|
|
|
@*/
|
|
|
|
|
|
|
|
Ox.doc = (function() {
|
|
|
|
// fixme: dont require the trailing '@'
|
|
|
|
var re = {
|
2012-05-22 23:17:17 +00:00
|
|
|
item: /^(.+?) <(.+?)> (.+?)$/,
|
2011-10-07 01:04:47 +00:00
|
|
|
multiline: /^\/\*\@.*?\n([\w\W]+)\n.*?\@\*\/$/,
|
|
|
|
script: /\n(\s*<script>s*\n[\w\W]+\n\s*<\/script>s*)/g,
|
|
|
|
singleline: /^\/\/@\s*(.*?)\s*$/,
|
|
|
|
test: /\n(\s*> .+\n.+?)/g,
|
|
|
|
usage: /\(.*?\)/
|
|
|
|
},
|
|
|
|
types = {
|
|
|
|
a: 'array', b: 'boolean', d: 'date',
|
|
|
|
e: 'element', f: 'function', n: 'number',
|
|
|
|
o: 'object', r: 'regexp', s: 'string',
|
2012-04-07 10:47:40 +00:00
|
|
|
u: 'undefined', '*': 'value', '!': 'event'
|
2011-10-07 01:04:47 +00:00
|
|
|
};
|
|
|
|
function decodeLinebreaks(match, submatch) {
|
|
|
|
return (submatch || match).replace(/\u21A9/g, '\n');
|
|
|
|
}
|
|
|
|
function encodeLinebreaks(match, submatch) {
|
|
|
|
return '\n' + (submatch || match).replace(/\n/g, '\u21A9');
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function getIndent(string) {
|
2011-10-07 01:04:47 +00:00
|
|
|
var indent = -1;
|
2012-05-25 11:42:25 +00:00
|
|
|
while (string[++indent] == ' ') {}
|
2011-10-07 01:04:47 +00:00
|
|
|
return indent;
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function parseItem(string) {
|
|
|
|
var matches = re.item.exec(string);
|
2011-10-07 01:04:47 +00:00
|
|
|
// to tell a variable with default value, like
|
|
|
|
// name <string|'<a href="...">foo</a>'> summary
|
|
|
|
// from a line of description with tags, like
|
|
|
|
// some <a href="...">description</a> text
|
|
|
|
// we need to check if there is either no forward slash
|
|
|
|
// or if the second last char is a single or double quote
|
|
|
|
return matches && (
|
|
|
|
matches[2].indexOf('/') == -1 ||
|
2012-05-24 09:47:33 +00:00
|
|
|
'\'"'.indexOf(matches[2].slice(-2, -1)) > -1
|
2011-10-07 01:04:47 +00:00
|
|
|
) ? Ox.extend({
|
|
|
|
name: parseName(matches[1].trim()),
|
|
|
|
summary: matches[3].trim()
|
|
|
|
}, parseType(matches[2])) : null;
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function parseName(string) {
|
|
|
|
var matches = re.usage.exec(string);
|
|
|
|
return matches ? matches[0] : string;
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
function parseNode(node) {
|
|
|
|
var item = parseItem(node.line), subitem;
|
|
|
|
node.nodes && node.nodes.forEach(function(node) {
|
|
|
|
var key, line = node.line, subitem;
|
|
|
|
if (!/^#/.test(node.line)) {
|
|
|
|
if (/^<script>/.test(line)) {
|
|
|
|
item.examples = [parseScript(line)];
|
|
|
|
} else if (/^>/.test(line)) {
|
|
|
|
item.examples = item.examples || [];
|
|
|
|
item.examples.push(parseTest(line));
|
|
|
|
} else if ((subitem = parseItem(line))) {
|
|
|
|
if (/^\(/.test(subitem.name)) {
|
|
|
|
item.usage = item.usage || [];
|
|
|
|
item.usage.push(parseNode(node));
|
|
|
|
} else if (subitem.types[0] == 'event') {
|
|
|
|
item.events = item.events || [];
|
|
|
|
item.events.push(parseNode(node));
|
|
|
|
} else {
|
|
|
|
key = item.types[0] == 'function'
|
|
|
|
? 'arguments' : 'properties'
|
|
|
|
item[key] = item[key] || [];
|
|
|
|
item[key].push(parseNode(node));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
item.description = item.description
|
|
|
|
? item.description + ' ' + line : line
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return item;
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function parseScript(string) {
|
2011-10-07 01:04:47 +00:00
|
|
|
// remove script tags and extra indentation
|
2012-05-25 11:42:25 +00:00
|
|
|
var lines = decodeLinebreaks(string).split('\n'),
|
2011-10-07 01:04:47 +00:00
|
|
|
indent = getIndent(lines[1]);
|
|
|
|
return {
|
2012-05-22 17:50:26 +00:00
|
|
|
statement: lines.slice(1, -1).map(function(line, i) {
|
|
|
|
return line.slice(indent);
|
2011-10-07 01:04:47 +00:00
|
|
|
}).join('\n')
|
|
|
|
};
|
|
|
|
}
|
2012-04-07 14:40:07 +00:00
|
|
|
function parseSource(source, file) {
|
|
|
|
var blocks = [],
|
|
|
|
items = [],
|
|
|
|
section = '',
|
|
|
|
tokens = [];
|
|
|
|
Ox.tokenize(source).forEach(function(token) {
|
|
|
|
var match;
|
|
|
|
token.source = source.substr(token.offset, token.length);
|
2012-04-09 08:39:02 +00:00
|
|
|
if (token.type == 'comment' && (
|
|
|
|
match = re.multiline.exec(token.source)
|
|
|
|
|| re.singleline.exec(token.source)
|
2012-04-07 14:40:07 +00:00
|
|
|
)) {
|
|
|
|
blocks.push(match[1]);
|
|
|
|
tokens.push([]);
|
|
|
|
} else if (tokens.length) {
|
|
|
|
tokens[tokens.length - 1].push(token);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
blocks.forEach(function(block, i) {
|
|
|
|
var item, lastItem,
|
|
|
|
lines = block
|
|
|
|
.replace(re.script, encodeLinebreaks)
|
|
|
|
.replace(re.test, encodeLinebreaks)
|
|
|
|
.split('\n'),
|
|
|
|
tree = parseTree(lines);
|
|
|
|
if (re.item.test(tree.line)) {
|
|
|
|
// parse the tree's root node
|
|
|
|
item = parseNode(tree);
|
|
|
|
item.file = file || '';
|
|
|
|
if (section) {
|
|
|
|
item.section = section;
|
|
|
|
}
|
|
|
|
if (/^[A-Z]/.test(item.name)) {
|
|
|
|
// main item
|
|
|
|
// include leading whitespace
|
|
|
|
item.source = parseTokens(tokens[i]);
|
2012-05-24 09:47:33 +00:00
|
|
|
item.line = source.slice(0, item.source[0].offset)
|
2012-04-07 14:40:07 +00:00
|
|
|
.split('\n').length;
|
|
|
|
items.push(item);
|
|
|
|
} else {
|
|
|
|
// property of a function item
|
|
|
|
lastItem = items[items.length - 1];
|
|
|
|
lastItem.properties = lastItem.properties || [];
|
|
|
|
lastItem.properties.push(item);
|
|
|
|
// include leading linebreaks and whitespace
|
2012-05-24 07:45:33 +00:00
|
|
|
lastItem.source = lastItem.source.concat(
|
|
|
|
parseTokens(tokens[i], true)
|
2012-04-07 14:40:07 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
section = tree.line.split(' ')[0]
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return items;
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function parseTest(string) {
|
2011-10-07 01:04:47 +00:00
|
|
|
// fixme: we cannot properly handle tests where a string contains '\n '
|
2012-05-25 11:42:25 +00:00
|
|
|
var lines = decodeLinebreaks(string).split('\n ');
|
2011-10-07 01:04:47 +00:00
|
|
|
return {
|
2012-05-24 09:47:33 +00:00
|
|
|
statement: lines[0].slice(2),
|
2011-10-07 01:04:47 +00:00
|
|
|
result: lines[1].trim()
|
|
|
|
};
|
|
|
|
}
|
2011-10-06 01:56:04 +00:00
|
|
|
function parseTokens(tokens, includeLeadingLinebreaks) {
|
|
|
|
var isLeading = true,
|
|
|
|
isTrailing = false,
|
|
|
|
tokens_ = [],
|
|
|
|
types = ['linebreak', 'whitespace'];
|
2011-10-07 01:04:47 +00:00
|
|
|
tokens.forEach(function(token) {
|
2011-10-06 01:56:04 +00:00
|
|
|
if (isLeading && types.indexOf(token.type) > -1) {
|
|
|
|
if (token.type == 'linebreak') {
|
|
|
|
if (includeLeadingLinebreaks) {
|
|
|
|
tokens_.push(token);
|
|
|
|
} else {
|
|
|
|
tokens_ = [];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tokens_.push(token);
|
|
|
|
}
|
2011-10-07 01:04:47 +00:00
|
|
|
} else {
|
2011-10-06 01:56:04 +00:00
|
|
|
tokens_.push(token);
|
|
|
|
isLeading = false;
|
|
|
|
if (types.indexOf(token.type) == -1) {
|
|
|
|
isTrailing = true;
|
|
|
|
}
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
});
|
2011-10-06 01:56:04 +00:00
|
|
|
if (isTrailing) {
|
|
|
|
while (types.indexOf(tokens_[tokens_.length - 1].type) > -1) {
|
|
|
|
tokens_.pop();
|
|
|
|
}
|
|
|
|
}
|
2011-10-07 01:04:47 +00:00
|
|
|
return tokens_;
|
|
|
|
}
|
|
|
|
function parseTree(lines) {
|
|
|
|
// parses indented lines into a tree structure, like
|
|
|
|
// {line: "...", nodes: [{line: "...", nodes: [...]}]}
|
|
|
|
var branches = [],
|
|
|
|
indent,
|
|
|
|
node = {
|
|
|
|
// chop the root line
|
|
|
|
line: lines.shift().trim()
|
|
|
|
};
|
|
|
|
if (lines.length) {
|
|
|
|
indent = getIndent(lines[0]);
|
|
|
|
lines.forEach(function(line) {
|
|
|
|
if (getIndent(line) == indent) {
|
|
|
|
// line is a child,
|
|
|
|
// make it the root line of a new branch
|
|
|
|
branches.push([line]);
|
|
|
|
} else {
|
|
|
|
// line is a descendant of the last child,
|
|
|
|
// add it to the last branch
|
|
|
|
branches[branches.length - 1].push(line);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
node.nodes = branches.map(function(lines) {
|
|
|
|
return parseTree(lines);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function parseType(string) {
|
2011-10-07 01:04:47 +00:00
|
|
|
// returns {types: [""]}
|
|
|
|
// or {types: [""], default: ""}
|
2012-04-09 08:39:02 +00:00
|
|
|
// or {types: [""], super: ""}
|
2012-05-25 11:42:25 +00:00
|
|
|
var array,
|
|
|
|
isArray,
|
2011-10-07 01:04:47 +00:00
|
|
|
ret = {types: []},
|
|
|
|
type;
|
|
|
|
// only split by ':' if there is no default string value
|
2012-05-25 11:42:25 +00:00
|
|
|
if ('\'"'.indexOf(string.slice(-2, -1)) == -1) {
|
|
|
|
array = string.split(':');
|
|
|
|
string = array[0];
|
|
|
|
if (array.length == 2) {
|
2012-05-25 16:28:05 +00:00
|
|
|
ret['super'] = array[1];
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
string.split('|').forEach(function(string) {
|
|
|
|
var unwrapped = unwrap(string);
|
2011-10-07 01:04:47 +00:00
|
|
|
if (unwrapped in types) {
|
|
|
|
ret.types.push(wrap(types[unwrapped]))
|
|
|
|
} else if (
|
|
|
|
(type = Ox.filter(Ox.values(types), function(type) {
|
|
|
|
return Ox.startsWith(type, unwrapped);
|
|
|
|
})).length
|
|
|
|
) {
|
|
|
|
ret.types.push(wrap(type[0]));
|
|
|
|
} else {
|
2012-05-25 11:42:25 +00:00
|
|
|
ret['default'] = string;
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
});
|
2012-05-25 11:42:25 +00:00
|
|
|
function unwrap(string) {
|
|
|
|
return (isArray = /^\[.+\]$/.test(string))
|
|
|
|
? string.slice(1, -1) : string;
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
2012-05-25 11:42:25 +00:00
|
|
|
function wrap(string) {
|
|
|
|
return isArray ? '[' + string + 's' + ']' : string;
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2012-04-09 08:39:02 +00:00
|
|
|
return function(/* source | file, callback | files, callback*/) {
|
|
|
|
var source = arguments.length == 1 ? arguments[0] : void 0,
|
2012-05-19 08:40:59 +00:00
|
|
|
files = arguments.length == 2 ? Ox.makeArray(arguments[0]) : void 0,
|
2012-04-09 08:39:02 +00:00
|
|
|
callback = arguments[1],
|
|
|
|
counter = 0, items = [];
|
|
|
|
files && files.forEach(function(file) {
|
2012-04-07 14:40:07 +00:00
|
|
|
Ox.get(file, function(source) {
|
2012-05-24 07:45:33 +00:00
|
|
|
items = items.concat(parseSource(source, file));
|
2012-04-09 08:39:02 +00:00
|
|
|
++counter == files.length && callback(items);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return source ? parseSource(source) : void 0;
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
}());
|
|
|
|
|
|
|
|
/*@
|
|
|
|
Ox.minify <f> Minifies JavaScript
|
|
|
|
(source) -> <s> Minified JavaScript
|
|
|
|
(file, callback) -> <u> undefined
|
|
|
|
source <s> JavaScript source
|
|
|
|
file <s> JavaScript file
|
|
|
|
callback <f> Callback function
|
|
|
|
> Ox.minify('for (a in b)\n{\t\tc = void 0;\n}')
|
|
|
|
'for(a in b)\n{c=void 0;}'
|
|
|
|
> Ox.minify('return a; return 0; return "";')
|
|
|
|
'return a;return 0;return"";'
|
|
|
|
> Ox.minify('return\na;\nreturn\n0;\nreturn\n"";')
|
|
|
|
'return\na;\nreturn\n0;\nreturn\n"";\n'
|
|
|
|
@*/
|
|
|
|
Ox.minify = function() {
|
|
|
|
// see https://github.com/douglascrockford/JSMin/blob/master/README
|
|
|
|
// and http://inimino.org/~inimino/blog/javascript_semicolons
|
|
|
|
if (arguments.length == 1) {
|
|
|
|
return minify(arguments[0]);
|
|
|
|
} else {
|
|
|
|
Ox.get(arguments[0], function(source) {
|
|
|
|
arguments[1](minify(source));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
function minify(source) {
|
|
|
|
var tokens = Ox.tokenize(source),
|
|
|
|
length = tokens.length;
|
|
|
|
function isIdentifierOrNumber(token) {
|
|
|
|
return [
|
|
|
|
'constant', 'identifier', 'keyword',
|
|
|
|
'number', 'method', 'object', 'property'
|
|
|
|
].indexOf(token.type) > -1;
|
|
|
|
}
|
2012-05-22 14:29:37 +00:00
|
|
|
return Ox.filter(Ox.map(tokens, function(token, i) {
|
2011-10-07 01:04:47 +00:00
|
|
|
var ret = source.substr(token.offset, token.length);
|
|
|
|
if (token.type == 'comment') {
|
|
|
|
if (ret[1] == '/') {
|
|
|
|
// replace line comment with newline
|
|
|
|
ret = i == length - 1 || tokens[i + 1].type == 'linebreak'
|
|
|
|
? null : '\n';
|
|
|
|
} else {
|
|
|
|
// replace block comment with space
|
|
|
|
ret = i == length - 1 || tokens[i + 1].type == 'whitespace'
|
|
|
|
? null : ' ';
|
|
|
|
}
|
2012-05-22 14:29:37 +00:00
|
|
|
} else if (token.type == 'linebreak' || token.type == 'whitespace') {
|
2011-10-07 01:04:47 +00:00
|
|
|
// remove consecutive linebreaks or whitespace
|
|
|
|
ret = ret[0];
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
// strip linebreaks, except between two tokens that
|
|
|
|
// are both either identifier or number or string
|
|
|
|
// or identifier in brackets or identifier with
|
|
|
|
// unary operator or array literal or object literal
|
|
|
|
//
|
|
|
|
// strip linebreaks, except after "break", "continue",
|
|
|
|
// "return", "++" or "--", where the linebreak becomes
|
|
|
|
// a semicolon
|
|
|
|
token.type == 'linebreak' && (
|
|
|
|
i == 0 || i == length - 1
|
|
|
|
|| (
|
|
|
|
!isIdentifierOrNumber(tokens[i - 1])
|
|
|
|
&& !tokens[i - 1].type == 'string'
|
|
|
|
&& [')', ']', '}', '++', '--'].indexOf(
|
|
|
|
source.substr(tokens[i - 1].offset, tokens[i - 1].length)
|
|
|
|
) == -1
|
|
|
|
)
|
|
|
|
|| (
|
|
|
|
!isIdentifierOrNumber(tokens[i + 1])
|
|
|
|
&& ['(', '[', '{', '+', '-', '++', '--'].indexOf(
|
|
|
|
source.substr(tokens[i + 1].offset, tokens[i + 1].length)
|
|
|
|
) == -1
|
|
|
|
)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
ret = null;
|
|
|
|
} else if (
|
|
|
|
// strip whitespace, except between two tokens
|
|
|
|
// that are both either identifier or number
|
|
|
|
token.type == 'whitespace' && (
|
|
|
|
i == 0 || i == length - 1
|
|
|
|
|| !isIdentifierOrNumber(tokens[i - 1])
|
|
|
|
|| !isIdentifierOrNumber(tokens[i + 1])
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
ret = null;
|
|
|
|
}
|
|
|
|
return ret;
|
2012-05-22 14:29:37 +00:00
|
|
|
})).join('');
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*@
|
|
|
|
Ox.test <f> Takes JavaScript, runs inline tests, returns results
|
|
|
|
@*/
|
|
|
|
Ox.test = function(file, callback) {
|
|
|
|
Ox.doc(file, function(items) {
|
|
|
|
var tests = [];
|
|
|
|
items.forEach(function(item) {
|
2012-05-25 16:32:54 +00:00
|
|
|
item.examples.some(function(example) {
|
|
|
|
return example.result;
|
|
|
|
}) && item.examples.forEach(function(example) {
|
2012-05-24 17:10:51 +00:00
|
|
|
Ox.Log('TEST', example.statement);
|
2011-10-07 01:04:47 +00:00
|
|
|
var actual = eval(example.statement);
|
|
|
|
if (example.result) {
|
|
|
|
tests.push({
|
|
|
|
actual: JSON.stringify(actual),
|
|
|
|
expected: example.result,
|
|
|
|
name: item.name,
|
|
|
|
section: item.section,
|
|
|
|
statement: example.statement,
|
|
|
|
passed: Ox.isEqual(eval(
|
|
|
|
'Ox.test.result = ' + example.result
|
|
|
|
), actual)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
callback(tests);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/*@
|
|
|
|
Ox.tokenize <f> Tokenizes JavaScript
|
|
|
|
(source) -> <[o]> Array of tokens
|
|
|
|
length <n> Length of the token
|
|
|
|
offset <n> Offset of the token
|
|
|
|
type <s> Type of the token
|
|
|
|
Type can be <code>"comment"</code>, <code>"constant"</code>,
|
|
|
|
<code>"identifier"</code>, <code>"keyword"</code>,
|
|
|
|
<code>"linebreak"</code>, <code>"method"</code>,
|
|
|
|
<code>"number"</code>, <code>"object"</code>,
|
|
|
|
<code>"operator"</code>, <code>"property"</code>,
|
|
|
|
<code>"regexp"</code>, <code>"string"</code>
|
|
|
|
or <code>"whitespace"</code>
|
|
|
|
source <s> JavaScript source code
|
|
|
|
@*/
|
2012-04-09 08:39:02 +00:00
|
|
|
// FIXME: constant/method/object/property is of interest
|
|
|
|
// for syntax highlighting, but may not belong here
|
2011-10-07 01:04:47 +00:00
|
|
|
Ox.tokenize = (function() {
|
|
|
|
|
|
|
|
// see https://github.com/mozilla/narcissus/blob/master/lib/jslex.js
|
|
|
|
// and https://developer.mozilla.org/en/JavaScript/Reference
|
|
|
|
|
|
|
|
var identifier = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ$_',
|
|
|
|
linebreak = '\n\r',
|
|
|
|
number = '0123456789',
|
|
|
|
operator = [
|
|
|
|
// arithmetic
|
|
|
|
'+', '-', '*', '/', '%', '++', '--',
|
|
|
|
// assignment
|
|
|
|
'=', '+=', '-=', '*=', '/=', '%=',
|
|
|
|
'&=', '|=', '^=', '<<=', '>>=', '>>>=',
|
|
|
|
// bitwise
|
|
|
|
'&', '|', '^', '~', '<<', '>>', '>>>',
|
|
|
|
// comparison
|
|
|
|
'==', '!=', '===', '!==', '>', '>=', '<', '<=',
|
|
|
|
// conditional
|
|
|
|
'?', ':',
|
|
|
|
// grouping
|
|
|
|
'(', ')', '[', ']', '{', '}',
|
|
|
|
// logical
|
|
|
|
'&&', '||', '!',
|
|
|
|
// other
|
|
|
|
'.', ',', ';'
|
|
|
|
],
|
2011-10-31 14:14:54 +00:00
|
|
|
regexp = 'abcdefghijklmnopqrstuvwxyz',
|
2011-10-07 01:04:47 +00:00
|
|
|
string = '\'"',
|
|
|
|
whitespace = ' \t',
|
|
|
|
word = {
|
|
|
|
constant: [
|
|
|
|
// Math
|
|
|
|
'E', 'LN2', 'LN10', 'LOG2E', 'LOG10E', 'PI', 'SQRT1_2', 'SQRT2',
|
|
|
|
// Number
|
|
|
|
'MAX_VALUE', 'MIN_VALUE', 'NEGATIVE_INFINITY', 'POSITIVE_INFINITY'
|
|
|
|
],
|
|
|
|
keyword: [
|
|
|
|
'break',
|
|
|
|
'case', 'catch', 'class', 'const', 'continue',
|
|
|
|
'debugger', 'default', 'delete', 'do',
|
|
|
|
'else', 'enum', 'export', 'extends',
|
|
|
|
'false', 'finally', 'for', 'function',
|
|
|
|
'if', 'implements', 'import', 'in', 'instanceof', 'interface',
|
|
|
|
'let', 'module',
|
|
|
|
'new', 'null',
|
|
|
|
'package', 'private', 'protected', 'public',
|
|
|
|
'return',
|
|
|
|
'super', 'switch', 'static',
|
|
|
|
'this', 'throw', 'true', 'try', 'typeof',
|
|
|
|
'var', 'void',
|
|
|
|
'yield',
|
|
|
|
'while', 'with',
|
|
|
|
],
|
|
|
|
method: [
|
|
|
|
// Array
|
|
|
|
'concat',
|
|
|
|
'every',
|
|
|
|
'filter', 'forEach',
|
|
|
|
'join',
|
|
|
|
'lastIndexOf',
|
|
|
|
'indexOf', 'isArray',
|
|
|
|
'map',
|
|
|
|
'pop', 'push',
|
|
|
|
'reduce', 'reduceRight', 'reverse',
|
|
|
|
'shift', 'slice', 'some', 'sort', 'splice',
|
|
|
|
'unshift',
|
|
|
|
// Date
|
|
|
|
'getDate', 'getDay', 'getFullYear', 'getHours',
|
|
|
|
'getMilliseconds', 'getMinutes', 'getMonth', 'getSeconds',
|
|
|
|
'getTime', 'getTimezoneOffset',
|
|
|
|
'getUTCDate', 'getUTCDay', 'getUTCFullYear', 'getUTCHours',
|
|
|
|
'getUTCMilliseconds', 'getUTCMinutes', 'getUTCMonth', 'getUTCSeconds',
|
|
|
|
'now',
|
|
|
|
'parse',
|
|
|
|
'setDate', 'setFullYear', 'setHours', 'setMilliseconds',
|
|
|
|
'setMinutes', 'setMonth', 'setSeconds', 'setTime',
|
|
|
|
'setUTCDate', 'setUTCFullYear', 'setUTCHours', 'setUTCMilliseconds',
|
|
|
|
'setUTCMinutes', 'setUTCMonth', 'setUTCSeconds',
|
|
|
|
'toDateString', 'toJSON', 'toLocaleDateString', 'toLocaleString',
|
|
|
|
'toLocaleTimeString', 'toTimeString', 'toUTCString',
|
|
|
|
'UTC',
|
|
|
|
// Function
|
|
|
|
'apply', 'bind', 'call', 'isGenerator',
|
|
|
|
// JSON
|
|
|
|
'parse', 'stringify',
|
|
|
|
// Math
|
|
|
|
'abs', 'acos', 'asin', 'atan', 'atan2',
|
|
|
|
'ceil', 'cos',
|
|
|
|
'exp',
|
|
|
|
'floor',
|
|
|
|
'log',
|
|
|
|
'max', 'min',
|
|
|
|
'pow',
|
|
|
|
'random', 'round',
|
|
|
|
'sin', 'sqrt',
|
|
|
|
'tan',
|
|
|
|
// Number
|
|
|
|
'toExponential', 'toFixed', 'toLocaleString', 'toPrecision',
|
|
|
|
// Object
|
|
|
|
'create',
|
|
|
|
'defineProperty', 'defineProperties',
|
|
|
|
'freeze',
|
|
|
|
'getOwnPropertyDescriptor', 'getOwnPropertyNames', 'getPrototypeOf',
|
|
|
|
'hasOwnProperty',
|
|
|
|
'isExtensible', 'isFrozen', 'isPrototypeOf', 'isSealed',
|
|
|
|
'keys',
|
|
|
|
'preventExtensions', 'propertyIsEnumerable',
|
|
|
|
'seal',
|
|
|
|
'toLocaleString', 'toString',
|
|
|
|
'valueOf',
|
|
|
|
// RegExp
|
|
|
|
'exec', 'test',
|
|
|
|
// String
|
|
|
|
'charAt', 'charCodeAt', 'concat',
|
|
|
|
'fromCharCode',
|
|
|
|
'indexOf',
|
|
|
|
'lastIndexOf', 'localeCompare',
|
|
|
|
'match',
|
|
|
|
'replace',
|
|
|
|
'search', 'slice', 'split', 'substr', 'substring',
|
|
|
|
'toLocaleLowerCase', 'toLocaleUpperCase', 'toLowerCase', 'toUpperCase', 'trim',
|
|
|
|
// Window
|
|
|
|
'addEventListener', 'alert', 'atob',
|
|
|
|
'blur', 'btoa',
|
|
|
|
'clearInterval', 'clearTimeout', 'close', 'confirm',
|
|
|
|
'dispatchEvent',
|
|
|
|
'escape',
|
|
|
|
'find', 'focus',
|
|
|
|
'getComputedStyle', 'getSelection',
|
|
|
|
'moveBy', 'moveTo',
|
|
|
|
'open',
|
|
|
|
'postMessage', 'print', 'prompt',
|
|
|
|
'removeEventListener', 'resizeBy', 'resizeTo',
|
|
|
|
'scroll', 'scrollBy', 'scrollTo',
|
|
|
|
'setCursor', 'setInterval', 'setTimeout', 'stop',
|
|
|
|
'unescape'
|
|
|
|
],
|
|
|
|
object: [
|
|
|
|
'Array',
|
|
|
|
'Boolean',
|
|
|
|
'Date', 'decodeURI', 'decodeURIComponent',
|
|
|
|
'encodeURI', 'encodeURIComponent', 'Error', 'eval', 'EvalError',
|
|
|
|
'Function',
|
|
|
|
'Infinity', 'isFinite', 'isNaN',
|
|
|
|
'JSON',
|
|
|
|
'Math',
|
|
|
|
'NaN', 'Number',
|
|
|
|
'Object',
|
|
|
|
'parseFloat', 'parseInt',
|
|
|
|
'RangeError', 'ReferenceError', 'RegExp',
|
|
|
|
'String', 'SyntaxError',
|
|
|
|
'TypeError',
|
|
|
|
'undefined', 'URIError',
|
|
|
|
'window'
|
|
|
|
],
|
|
|
|
property: [
|
|
|
|
// Function
|
|
|
|
'constructor', 'length', 'prototype',
|
|
|
|
// RegExp
|
|
|
|
'global', 'ignoreCase', 'lastIndex', 'multiline', 'source',
|
|
|
|
// Window
|
|
|
|
'applicationCache',
|
|
|
|
'closed', 'console', 'content', 'crypto',
|
|
|
|
'defaultStatus', 'document',
|
|
|
|
'frameElement', 'frames',
|
|
|
|
'history',
|
|
|
|
'innerHeight', 'innerWidth',
|
|
|
|
'length', 'location', 'locationbar', 'localStorage',
|
|
|
|
'menubar',
|
|
|
|
'name', 'navigator',
|
|
|
|
'opener', 'outerHeight', 'outerWidth',
|
|
|
|
'pageXOffset', 'pageYOffset', 'parent', 'personalbar',
|
|
|
|
'screen', 'screenX', 'screenY', 'scrollbars', 'scrollX', 'scrollY',
|
|
|
|
'self', 'sessionStorage', 'status', 'statusbar',
|
|
|
|
'toolbar', 'top'
|
|
|
|
]
|
|
|
|
};
|
|
|
|
|
|
|
|
return function(source) {
|
|
|
|
|
|
|
|
source = source.replace(/\r\n/g, '\n').replace(/\r/g, '\n');
|
|
|
|
|
|
|
|
var cursor = 0,
|
|
|
|
tokenize = {
|
|
|
|
comment: function() {
|
|
|
|
while (char = source[++cursor]) {
|
|
|
|
if (next == '/' && char == '\n') {
|
|
|
|
break;
|
|
|
|
} else if (next == '*' && char + source[cursor + 1] == '*/') {
|
|
|
|
cursor += 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
identifier: function() {
|
|
|
|
var str;
|
|
|
|
while ((identifier + number).indexOf(source[++cursor]) > -1) {}
|
2012-05-24 09:47:33 +00:00
|
|
|
str = source.slice(start, cursor);
|
2011-10-07 01:04:47 +00:00
|
|
|
Ox.forEach(word, function(value, key) {
|
|
|
|
if (value.indexOf(str) > -1) {
|
|
|
|
type = key;
|
2012-05-25 07:32:32 +00:00
|
|
|
Ox.Break();
|
2011-10-07 01:04:47 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
linebreak: function() {
|
|
|
|
while (linebreak.indexOf(source[++cursor]) > -1) {}
|
|
|
|
},
|
|
|
|
number: function() {
|
|
|
|
while ((number + '.').indexOf(source[++cursor]) > -1) {}
|
|
|
|
},
|
|
|
|
operator: function() {
|
|
|
|
while (operator.indexOf(char += source[++cursor]) > -1) {}
|
|
|
|
},
|
|
|
|
regexp: function() {
|
|
|
|
while ((char = source[++cursor]) != '/') {
|
|
|
|
char == '\\' && ++cursor;
|
|
|
|
if (cursor == source.length) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (regexp.indexOf(source[++cursor]) > -1) {}
|
|
|
|
},
|
|
|
|
string: function() {
|
|
|
|
var delimiter = char;
|
|
|
|
while ((char = source[++cursor]) != delimiter) {
|
|
|
|
char == '\\' && ++cursor;
|
|
|
|
if (cursor == source.length) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++cursor;
|
|
|
|
},
|
|
|
|
whitespace: function() {
|
|
|
|
while (whitespace.indexOf(source[++cursor]) > -1) {}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
tokens = [],
|
|
|
|
type;
|
|
|
|
|
|
|
|
while (cursor < source.length) {
|
|
|
|
var char = source[cursor],
|
|
|
|
next = source[cursor + 1],
|
|
|
|
start = cursor;
|
|
|
|
if (char == '/' && (next == '/' || next == '*')) {
|
|
|
|
type = 'comment';
|
|
|
|
} else if (identifier.indexOf(char) > -1) {
|
|
|
|
type = 'identifier';
|
|
|
|
} else if (linebreak.indexOf(char) > -1) {
|
|
|
|
type = 'linebreak';
|
|
|
|
} else if (number.indexOf(char) > -1) {
|
|
|
|
type = 'number';
|
|
|
|
} else if (string.indexOf(char) > -1) {
|
|
|
|
type = 'string';
|
|
|
|
} else if (whitespace.indexOf(char) > -1) {
|
|
|
|
type = 'whitespace';
|
|
|
|
} else if (char == '/') {
|
|
|
|
type = isRegExp() ? 'regexp' : 'operator';
|
|
|
|
} else if (operator.indexOf(char) > -1) {
|
|
|
|
type = 'operator';
|
|
|
|
}
|
|
|
|
tokenize[type]();
|
|
|
|
tokens.push({
|
|
|
|
length: cursor - start,
|
|
|
|
offset: start,
|
|
|
|
type: type,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function isRegExp() {
|
|
|
|
// checks if a forward slash is the beginning of a regexp,
|
|
|
|
// as opposed to the beginning of an operator
|
|
|
|
// see http://www.mozilla.org/js/language/js20-2000-07/rationale/syntax.html#regular-expressions
|
|
|
|
var index = tokens.length,
|
|
|
|
isRegExp = false,
|
|
|
|
offset = 0,
|
|
|
|
prevToken,
|
|
|
|
prevString;
|
|
|
|
// scan back to the previous significant token,
|
|
|
|
// or the beginning of the source
|
|
|
|
while (
|
2012-01-04 07:42:48 +00:00
|
|
|
tokens[--index] !== void 0 && [
|
|
|
|
'comment', 'linebreak', 'whitespace'
|
|
|
|
].indexOf(tokens[index].type) > -1
|
2011-10-07 01:04:47 +00:00
|
|
|
) {
|
|
|
|
offset += tokens[index].length;
|
|
|
|
}
|
|
|
|
if (typeof tokens[index] == 'undefined') {
|
|
|
|
// source begins with forward slash
|
|
|
|
isRegExp = true;
|
|
|
|
} else {
|
|
|
|
prevToken = tokens[index];
|
2012-01-04 07:42:48 +00:00
|
|
|
prevString = source.substr(
|
|
|
|
cursor - prevToken.length - offset, prevToken.length
|
|
|
|
);
|
2011-10-07 01:04:47 +00:00
|
|
|
isRegExp = (
|
|
|
|
prevToken.type == 'keyword'
|
|
|
|
&& ['false', 'null', 'true'].indexOf(prevString) == -1
|
|
|
|
) || (
|
|
|
|
prevToken.type == 'operator'
|
|
|
|
&& ['++', '--', ')', ']', '}'].indexOf(prevString) == -1
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return isRegExp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return tokens;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2011-10-31 14:14:54 +00:00
|
|
|
}());
|