CoffeeScript-in-CoffeeScript can compile dotted accessors

This commit is contained in:
Jeremy Ashkenas 2010-02-08 23:42:03 -05:00
parent 32098e5a13
commit cb57a1ca1f
4 changed files with 45 additions and 11 deletions

View File

@ -107,12 +107,12 @@
if (tag === 'IDENTIFIER' && this.value() === '::') { if (tag === 'IDENTIFIER' && this.value() === '::') {
this.tag(-1, 'PROTOTYPE_ACCESS'); this.tag(-1, 'PROTOTYPE_ACCESS');
} }
if (tag === 'IDENTIFIER' && this.value() === '.' && !(this.value(-2) === '.')) { if (tag === 'IDENTIFIER' && this.value() === '.' && !(this.value(2) === '.')) {
if (this.tag(-2) === '?') { if (this.tag(2) === '?') {
this.tag(-1, 'SOAK_ACCESS'); this.tag(1, 'SOAK_ACCESS');
this.tokens.splice(-2, 1); this.tokens.splice(-2, 1);
} else { } else {
this.tag(-1, 'PROPERTY_ACCESS'); this.tag(1, 'PROPERTY_ACCESS');
} }
} }
this.token(tag, id); this.token(tag, id);

View File

@ -1,5 +1,5 @@
(function(){ (function(){
var CallNode, CommentNode, Expressions, ExtendsNode, LiteralNode, Node, ReturnNode, TAB, TRAILING_WHITESPACE, ValueNode, any, compact, del, dup, flatten, inherit, merge, statement; var AccessorNode, CallNode, CommentNode, Expressions, ExtendsNode, LiteralNode, Node, ReturnNode, TAB, TRAILING_WHITESPACE, ValueNode, any, compact, del, dup, flatten, inherit, merge, statement;
var __hasProp = Object.prototype.hasOwnProperty; var __hasProp = Object.prototype.hasOwnProperty;
process.mixin(require('./scope')); process.mixin(require('./scope'));
// The abstract base class for all CoffeeScript nodes. // The abstract base class for all CoffeeScript nodes.
@ -553,7 +553,8 @@
}, },
push: function push(prop) { push: function push(prop) {
this.properties.push(prop); this.properties.push(prop);
return this.children.push(prop); this.children.push(prop);
return this;
}, },
has_properties: function has_properties() { has_properties: function has_properties() {
return this.properties.length || this.base instanceof ThisNode; return this.properties.length || this.base instanceof ThisNode;
@ -640,7 +641,8 @@
}, },
push: function push(arg) { push: function push(arg) {
this.args.push(arg); this.args.push(arg);
return this.children.push(arg); this.children.push(arg);
return this;
}, },
// Compile a vanilla function call. // Compile a vanilla function call.
compile_node: function compile_node(o) { compile_node: function compile_node(o) {
@ -718,4 +720,18 @@
} }
})); }));
statement(ExtendsNode); statement(ExtendsNode);
// A dotted accessor into a part of a value, or the :: shorthand for
// an accessor into the object's prototype.
AccessorNode = (exports.AccessorNode = inherit(Node, {
constructor: function constructor(name, tag) {
this.name = name;
this.children = [this.name];
this.prototype = tag === 'prototype';
this.soak = tag === 'soak';
return this;
},
compile_node: function compile_node(o) {
return '.' + (this.prototype ? 'prototype.' : '') + this.name.compile(o);
}
}));
})(); })();

View File

@ -96,12 +96,12 @@ lex::identifier_token: ->
tag: if KEYWORDS.indexOf(id) >= 0 then id.toUpperCase() else 'IDENTIFIER' tag: if KEYWORDS.indexOf(id) >= 0 then id.toUpperCase() else 'IDENTIFIER'
tag: 'LEADING_WHEN' if tag is 'WHEN' and (this.tag() is 'OUTDENT' or this.tag() is 'INDENT') tag: 'LEADING_WHEN' if tag is 'WHEN' and (this.tag() is 'OUTDENT' or this.tag() is 'INDENT')
this.tag(-1, 'PROTOTYPE_ACCESS') if tag is 'IDENTIFIER' and this.value() is '::' this.tag(-1, 'PROTOTYPE_ACCESS') if tag is 'IDENTIFIER' and this.value() is '::'
if tag is 'IDENTIFIER' and this.value() is '.' and !(this.value(-2) is '.') if tag is 'IDENTIFIER' and this.value() is '.' and !(this.value(2) is '.')
if this.tag(-2) is '?' if this.tag(2) is '?'
this.tag(-1, 'SOAK_ACCESS') this.tag(1, 'SOAK_ACCESS')
this.tokens.splice(-2, 1) this.tokens.splice(-2, 1)
else else
this.tag(-1, 'PROPERTY_ACCESS') this.tag(1, 'PROPERTY_ACCESS')
this.token(tag, id) this.token(tag, id)
this.i += id.length this.i += id.length
true true

View File

@ -290,6 +290,7 @@ ValueNode: exports.ValueNode: inherit Node, {
push: (prop) -> push: (prop) ->
@properties.push(prop) @properties.push(prop)
@children.push(prop) @children.push(prop)
this
has_properties: -> has_properties: ->
@properties.length or @base instanceof ThisNode @properties.length or @base instanceof ThisNode
@ -377,6 +378,7 @@ CallNode: exports.CallNode: inherit Node, {
push: (arg) -> push: (arg) ->
@args.push(arg) @args.push(arg)
@children.push(arg) @children.push(arg)
this
# Compile a vanilla function call. # Compile a vanilla function call.
compile_node: (o) -> compile_node: (o) ->
@ -441,6 +443,22 @@ ExtendsNode: exports.ExtendsNode: inherit Node, {
statement ExtendsNode statement ExtendsNode
# A dotted accessor into a part of a value, or the :: shorthand for
# an accessor into the object's prototype.
AccessorNode: exports.AccessorNode: inherit Node, {
constructor: (name, tag) ->
@name: name
@children: [@name]
@prototype: tag is 'prototype'
@soak: tag is 'soak'
this
compile_node: (o) ->
'.' + (if @prototype then 'prototype.' else '') + @name.compile(o)
}