You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
69 lines
3.2 KiB
69 lines
3.2 KiB
var jsp = require("./parse-js"), |
|
pro = require("./process"), |
|
slice = jsp.slice, |
|
member = jsp.member, |
|
curry = jsp.curry, |
|
MAP = pro.MAP, |
|
PRECEDENCE = jsp.PRECEDENCE, |
|
OPERATORS = jsp.OPERATORS; |
|
|
|
function ast_squeeze_more(ast) { |
|
var w = pro.ast_walker(), walk = w.walk, scope; |
|
function with_scope(s, cont) { |
|
var save = scope, ret; |
|
scope = s; |
|
ret = cont(); |
|
scope = save; |
|
return ret; |
|
}; |
|
function _lambda(name, args, body) { |
|
return [ this[0], name, args, with_scope(body.scope, curry(MAP, body, walk)) ]; |
|
}; |
|
return w.with_walkers({ |
|
"toplevel": function(body) { |
|
return [ this[0], with_scope(this.scope, curry(MAP, body, walk)) ]; |
|
}, |
|
"function": _lambda, |
|
"defun": _lambda, |
|
"new": function(ctor, args) { |
|
if (ctor[0] == "name") { |
|
if (ctor[1] == "Array" && !scope.has("Array")) { |
|
if (args.length != 1) { |
|
return [ "array", args ]; |
|
} else { |
|
return walk([ "call", [ "name", "Array" ], args ]); |
|
} |
|
} else if (ctor[1] == "Object" && !scope.has("Object")) { |
|
if (!args.length) { |
|
return [ "object", [] ]; |
|
} else { |
|
return walk([ "call", [ "name", "Object" ], args ]); |
|
} |
|
} else if ((ctor[1] == "RegExp" || ctor[1] == "Function" || ctor[1] == "Error") && !scope.has(ctor[1])) { |
|
return walk([ "call", [ "name", ctor[1] ], args]); |
|
} |
|
} |
|
}, |
|
"call": function(expr, args) { |
|
if (expr[0] == "dot" && expr[2] == "toString" && args.length == 0) { |
|
// foo.toString() ==> foo+"" |
|
return [ "binary", "+", expr[1], [ "string", "" ]]; |
|
} |
|
if (expr[0] == "name") { |
|
if (expr[1] == "Array" && args.length != 1 && !scope.has("Array")) { |
|
return [ "array", args ]; |
|
} |
|
if (expr[1] == "Object" && !args.length && !scope.has("Object")) { |
|
return [ "object", [] ]; |
|
} |
|
if (expr[1] == "String" && !scope.has("String")) { |
|
return [ "binary", "+", args[0], [ "string", "" ]]; |
|
} |
|
} |
|
} |
|
}, function() { |
|
return walk(pro.ast_add_scope(ast)); |
|
}); |
|
}; |
|
|
|
exports.ast_squeeze_more = ast_squeeze_more;
|
|
|