feat(transpiler): allow @CONST annotation on class

Closes #148
This commit is contained in:
Tommy Odom 2014-11-02 14:21:11 -05:00 committed by vsavkin
parent 0a766f4654
commit 8e6326f838
2 changed files with 61 additions and 2 deletions

View File

@ -21,6 +21,9 @@ class SubFoo extends Foo {
} }
} }
@CONST
class ConstClass {}
class Const { class Const {
@CONST @CONST
constructor(a:number) { constructor(a:number) {
@ -62,6 +65,11 @@ export function main() {
expect(subConst.b).toBe(2); expect(subConst.b).toBe(2);
}); });
it('@CONST on class without constructor should generate const constructor', function () {
var constClass = new ConstClass();
expect(constClass).not.toBe(null);
});
describe('inheritance', function() { describe('inheritance', function() {
it('should support super call', function () { it('should support super call', function () {
var subFoo = new SubFoo(1, 2); var subFoo = new SubFoo(1, 2);

View File

@ -19,6 +19,8 @@ import {propName} from 'traceur/src/staticsemantics/PropName';
import { import {
BinaryExpression, BinaryExpression,
BindingIdentifier, BindingIdentifier,
FormalParameterList,
FunctionBody,
IdentifierExpression IdentifierExpression
} from 'traceur/src/syntax/trees/ParseTrees'; } from 'traceur/src/syntax/trees/ParseTrees';
@ -48,6 +50,7 @@ export class ClassTransformer extends ParseTreeTransformer {
var argumentTypesMap = {}; var argumentTypesMap = {};
var fields = []; var fields = [];
var isConst; var isConst;
var hasConstructor = false;
var that = this; var that = this;
tree.elements.forEach(function(elementTree, index) { tree.elements.forEach(function(elementTree, index) {
@ -55,8 +58,8 @@ export class ClassTransformer extends ParseTreeTransformer {
!elementTree.isStatic && !elementTree.isStatic &&
propName(elementTree) === CONSTRUCTOR) { propName(elementTree) === CONSTRUCTOR) {
isConst = elementTree.annotations.some((annotation) => hasConstructor = true;
annotation.name.identifierToken.value === 'CONST'); isConst = elementTree.annotations.some(that._isConstAnnotation);
// Store constructor argument types, // Store constructor argument types,
// so that we can use them to set the types of simple-assigned fields. // so that we can use them to set the types of simple-assigned fields.
@ -124,6 +127,26 @@ export class ClassTransformer extends ParseTreeTransformer {
} }
}); });
// If no constructor exists then look to see if we should create a const constructor
// when the class is annotated with @CONST.
if (!hasConstructor) {
let constClassAnnotation = this._extractConstAnnotation(tree);
if (constClassAnnotation !== null) {
tree.elements = [(new PropertyConstructorAssignment(
constClassAnnotation.location,
false,
null,
tree.name,
new FormalParameterList(constClassAnnotation.location, []),
null,
[constClassAnnotation],
new FunctionBody(constClassAnnotation.location, []),
true,
[]
))].concat(tree.elements);
}
}
// Add the field definitions to the beginning of the class. // Add the field definitions to the beginning of the class.
tree.elements = fields.concat(tree.elements); tree.elements = fields.concat(tree.elements);
@ -183,4 +206,32 @@ export class ClassTransformer extends ParseTreeTransformer {
body.statements = statements; body.statements = statements;
return superCall; return superCall;
} }
/**
* Extract the @CONST annotation from the class annotations.
* When found the annotation is removed from the class annotations and returned.
*/
_extractConstAnnotation(tree) {
var annotations = [];
var constAnnotation = null;
var that = this;
tree.annotations.forEach((annotation) => {
if (that._isConstAnnotation(annotation)) {
constAnnotation = annotation;
} else {
annotations.push(annotation);
}
});
tree.annotations = annotations;
return constAnnotation;
}
/**
* Returns true if the annotation is @CONST
*/
_isConstAnnotation(annotation) {
return annotation.name.identifierToken.value === 'CONST';
}
} }