2016-06-23 12:47:54 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-06-23 12:47:54 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-12-27 12:36:47 -05:00
|
|
|
import {StaticSymbol} from '@angular/compiler/src/aot/static_symbol';
|
2016-04-28 20:50:03 -04:00
|
|
|
import * as o from '@angular/compiler/src/output/output_ast';
|
2016-08-02 18:53:34 -04:00
|
|
|
import {TypeScriptEmitter} from '@angular/compiler/src/output/ts_emitter';
|
2017-09-28 13:53:04 -04:00
|
|
|
import {ParseLocation, ParseSourceFile, ParseSourceSpan} from '@angular/compiler/src/parse_util';
|
2019-08-15 16:42:17 -04:00
|
|
|
import {newArray} from '@angular/compiler/src/util';
|
|
|
|
|
2017-03-14 12:16:15 -04:00
|
|
|
import {stripSourceMapAndNewLine} from './abstract_emitter_spec';
|
2017-01-27 17:23:12 -05:00
|
|
|
|
2017-03-15 18:50:30 -04:00
|
|
|
const someGenFilePath = 'somePackage/someGenFile';
|
2016-11-15 13:14:01 -05:00
|
|
|
const anotherModuleUrl = 'somePackage/someOtherPath';
|
2016-01-06 17:13:44 -05:00
|
|
|
|
2017-05-16 19:30:37 -04:00
|
|
|
const sameModuleIdentifier = new o.ExternalReference(null, 'someLocalId', null);
|
2016-01-06 17:13:44 -05:00
|
|
|
|
2017-05-16 19:30:37 -04:00
|
|
|
const externalModuleIdentifier = new o.ExternalReference(anotherModuleUrl, 'someExternalId', null);
|
2016-11-23 17:47:05 -05:00
|
|
|
|
2017-12-16 17:42:55 -05:00
|
|
|
{
|
2016-12-27 12:36:47 -05:00
|
|
|
// Not supported features of our OutputAst in TS:
|
2016-01-06 17:13:44 -05:00
|
|
|
// - real `const` like in Dart
|
|
|
|
// - final fields
|
|
|
|
|
|
|
|
describe('TypeScriptEmitter', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
let emitter: TypeScriptEmitter;
|
|
|
|
let someVar: o.ReadVarExpr;
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
beforeEach(() => {
|
2017-05-16 19:30:37 -04:00
|
|
|
emitter = new TypeScriptEmitter();
|
2017-03-24 12:59:58 -04:00
|
|
|
someVar = o.variable('someVar', null, null);
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2020-04-08 13:14:18 -04:00
|
|
|
function emitStmt(stmt: o.Statement|o.Statement[], preamble?: string): string {
|
2016-12-27 12:36:47 -05:00
|
|
|
const stmts = Array.isArray(stmt) ? stmt : [stmt];
|
2017-10-04 16:37:27 -04:00
|
|
|
const source = emitter.emitStatements(someGenFilePath, stmts, preamble);
|
2017-03-14 12:16:15 -04:00
|
|
|
return stripSourceMapAndNewLine(source);
|
2016-01-06 17:13:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
it('should declare variables', () => {
|
2016-05-07 11:20:56 -04:00
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toDeclStmt())).toEqual(`var someVar:any = 1;`);
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toDeclStmt(null, [o.StmtModifier.Final])))
|
2016-05-07 11:20:56 -04:00
|
|
|
.toEqual(`const someVar:any = 1;`);
|
2017-05-16 19:30:37 -04:00
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toDeclStmt(null, [o.StmtModifier.Exported])))
|
2016-05-07 11:20:56 -04:00
|
|
|
.toEqual(`export var someVar:any = 1;`);
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toDeclStmt(o.INT_TYPE)))
|
|
|
|
.toEqual(`var someVar:number = 1;`);
|
2017-02-17 11:56:49 -05:00
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toDeclStmt(o.INFERRED_TYPE)))
|
|
|
|
.toEqual(`var someVar = 1;`);
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2016-12-27 12:36:47 -05:00
|
|
|
describe('declare variables with ExternExpressions as values', () => {
|
|
|
|
it('should create no reexport if the identifier is in the same module', () => {
|
|
|
|
// identifier is in the same module -> no reexport
|
2017-05-16 19:30:37 -04:00
|
|
|
expect(emitStmt(someVar.set(o.importExpr(sameModuleIdentifier)).toDeclStmt(null, [
|
|
|
|
o.StmtModifier.Exported
|
|
|
|
]))).toEqual('export var someVar:any = someLocalId;');
|
2016-12-27 12:36:47 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should create no reexport if the variable is not exported', () => {
|
|
|
|
expect(emitStmt(someVar.set(o.importExpr(externalModuleIdentifier)).toDeclStmt())).toEqual([
|
2017-05-11 13:28:48 -04:00
|
|
|
`import * as i0 from 'somePackage/someOtherPath';`, `var someVar:any = i0.someExternalId;`
|
2016-12-27 12:36:47 -05:00
|
|
|
].join('\n'));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create no reexport if the variable is typed', () => {
|
2017-05-16 19:30:37 -04:00
|
|
|
expect(emitStmt(someVar.set(o.importExpr(externalModuleIdentifier))
|
|
|
|
.toDeclStmt(o.DYNAMIC_TYPE, [o.StmtModifier.Exported])))
|
2016-12-27 12:36:47 -05:00
|
|
|
.toEqual([
|
2017-05-11 13:28:48 -04:00
|
|
|
`import * as i0 from 'somePackage/someOtherPath';`,
|
|
|
|
`export var someVar:any = i0.someExternalId;`
|
2016-12-27 12:36:47 -05:00
|
|
|
].join('\n'));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create a reexport', () => {
|
2017-05-16 19:30:37 -04:00
|
|
|
expect(emitStmt(someVar.set(o.importExpr(externalModuleIdentifier))
|
|
|
|
.toDeclStmt(null, [o.StmtModifier.Exported])))
|
2016-12-27 12:36:47 -05:00
|
|
|
.toEqual([
|
|
|
|
`export {someExternalId as someVar} from 'somePackage/someOtherPath';`, ``
|
|
|
|
].join('\n'));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create multiple reexports from the same file', () => {
|
|
|
|
const someVar2 = o.variable('someVar2');
|
2017-05-16 19:30:37 -04:00
|
|
|
const externalModuleIdentifier2 =
|
|
|
|
new o.ExternalReference(anotherModuleUrl, 'someExternalId2', null);
|
|
|
|
expect(emitStmt([
|
|
|
|
someVar.set(o.importExpr(externalModuleIdentifier))
|
|
|
|
.toDeclStmt(null, [o.StmtModifier.Exported]),
|
|
|
|
someVar2.set(o.importExpr(externalModuleIdentifier2))
|
|
|
|
.toDeclStmt(null, [o.StmtModifier.Exported])
|
|
|
|
]))
|
2016-12-27 12:36:47 -05:00
|
|
|
.toEqual([
|
|
|
|
`export {someExternalId as someVar,someExternalId2 as someVar2} from 'somePackage/someOtherPath';`,
|
|
|
|
``
|
|
|
|
].join('\n'));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-01-06 17:13:44 -05:00
|
|
|
it('should read and write variables', () => {
|
|
|
|
expect(emitStmt(someVar.toStmt())).toEqual(`someVar;`);
|
|
|
|
expect(emitStmt(someVar.set(o.literal(1)).toStmt())).toEqual(`someVar = 1;`);
|
|
|
|
expect(emitStmt(someVar.set(o.variable('someOtherVar').set(o.literal(1))).toStmt()))
|
|
|
|
.toEqual(`someVar = (someOtherVar = 1);`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should read and write keys', () => {
|
|
|
|
expect(emitStmt(o.variable('someMap').key(o.variable('someKey')).toStmt()))
|
|
|
|
.toEqual(`someMap[someKey];`);
|
|
|
|
expect(emitStmt(o.variable('someMap').key(o.variable('someKey')).set(o.literal(1)).toStmt()))
|
|
|
|
.toEqual(`someMap[someKey] = 1;`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should read and write properties', () => {
|
|
|
|
expect(emitStmt(o.variable('someObj').prop('someProp').toStmt()))
|
|
|
|
.toEqual(`someObj.someProp;`);
|
|
|
|
expect(emitStmt(o.variable('someObj').prop('someProp').set(o.literal(1)).toStmt()))
|
|
|
|
.toEqual(`someObj.someProp = 1;`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should invoke functions and methods and constructors', () => {
|
|
|
|
expect(emitStmt(o.variable('someFn').callFn([o.literal(1)]).toStmt())).toEqual('someFn(1);');
|
|
|
|
expect(emitStmt(o.variable('someObj').callMethod('someMethod', [o.literal(1)]).toStmt()))
|
|
|
|
.toEqual('someObj.someMethod(1);');
|
|
|
|
expect(emitStmt(o.variable('SomeClass').instantiate([o.literal(1)]).toStmt()))
|
|
|
|
.toEqual('new SomeClass(1);');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support builtin methods', () => {
|
|
|
|
expect(emitStmt(o.variable('arr1')
|
|
|
|
.callMethod(o.BuiltinMethod.ConcatArray, [o.variable('arr2')])
|
|
|
|
.toStmt()))
|
|
|
|
.toEqual('arr1.concat(arr2);');
|
|
|
|
|
|
|
|
expect(emitStmt(o.variable('observable')
|
|
|
|
.callMethod(o.BuiltinMethod.SubscribeObservable, [o.variable('listener')])
|
|
|
|
.toStmt()))
|
|
|
|
.toEqual('observable.subscribe(listener);');
|
2016-04-22 18:33:32 -04:00
|
|
|
|
|
|
|
expect(
|
|
|
|
emitStmt(
|
2016-08-14 13:04:37 -04:00
|
|
|
o.variable('fn').callMethod(o.BuiltinMethod.Bind, [o.variable('someObj')]).toStmt()))
|
2016-04-22 18:33:32 -04:00
|
|
|
.toEqual('fn.bind(someObj);');
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support literals', () => {
|
|
|
|
expect(emitStmt(o.literal(0).toStmt())).toEqual('0;');
|
|
|
|
expect(emitStmt(o.literal(true).toStmt())).toEqual('true;');
|
|
|
|
expect(emitStmt(o.literal('someStr').toStmt())).toEqual(`'someStr';`);
|
|
|
|
expect(emitStmt(o.literalArr([o.literal(1)]).toStmt())).toEqual(`[1];`);
|
2017-07-05 17:51:39 -04:00
|
|
|
expect(emitStmt(o.literalMap([
|
|
|
|
{key: 'someKey', value: o.literal(1), quoted: false},
|
|
|
|
{key: 'a', value: o.literal('a'), quoted: false},
|
|
|
|
{key: '*', value: o.literal('star'), quoted: true},
|
|
|
|
]).toStmt())
|
|
|
|
.replace(/\s+/gm, ''))
|
|
|
|
.toEqual(`{someKey:1,a:'a','*':'star'};`);
|
2017-05-11 13:28:48 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should break expressions into multiple lines if they are too long', () => {
|
2019-08-15 16:42:17 -04:00
|
|
|
const values: o.Expression[] = newArray(100);
|
2017-05-11 13:28:48 -04:00
|
|
|
values.fill(o.literal(1));
|
|
|
|
values.splice(50, 0, o.fn([], [new o.ReturnStatement(o.literal(1))]));
|
|
|
|
expect(emitStmt(o.variable('fn').callFn(values).toStmt())).toEqual([
|
|
|
|
'fn(1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,',
|
|
|
|
' 1,1,1,1,1,1,1,1,1,1,():void => {', ' return 1;',
|
|
|
|
' },1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,',
|
|
|
|
' 1,1,1,1,1,1,1,1,1,1,1,1);'
|
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2016-11-04 13:55:21 -04:00
|
|
|
it('should support blank literals', () => {
|
|
|
|
expect(emitStmt(o.literal(null).toStmt())).toEqual('(null as any);');
|
|
|
|
expect(emitStmt(o.literal(undefined).toStmt())).toEqual('(undefined as any);');
|
2016-11-11 20:12:17 -05:00
|
|
|
expect(emitStmt(o.variable('a', null).isBlank().toStmt())).toEqual('(a == null);');
|
2016-11-04 13:55:21 -04:00
|
|
|
});
|
|
|
|
|
2016-01-06 17:13:44 -05:00
|
|
|
it('should support external identifiers', () => {
|
|
|
|
expect(emitStmt(o.importExpr(sameModuleIdentifier).toStmt())).toEqual('someLocalId;');
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(o.importExpr(externalModuleIdentifier).toStmt())).toEqual([
|
2017-05-11 13:28:48 -04:00
|
|
|
`import * as i0 from 'somePackage/someOtherPath';`, `i0.someExternalId;`
|
2016-06-08 19:38:52 -04:00
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support operators', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const lhs = o.variable('lhs');
|
|
|
|
const rhs = o.variable('rhs');
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(someVar.cast(o.INT_TYPE).toStmt())).toEqual('(<number>someVar);');
|
|
|
|
expect(emitStmt(o.not(someVar).toStmt())).toEqual('!someVar;');
|
2020-07-03 19:52:40 -04:00
|
|
|
expect(emitStmt(o.unary(o.UnaryOperator.Minus, someVar).toStmt())).toEqual('(-someVar);');
|
|
|
|
expect(emitStmt(o.unary(o.UnaryOperator.Plus, someVar).toStmt())).toEqual('(+someVar);');
|
2017-05-11 13:15:54 -04:00
|
|
|
expect(emitStmt(o.assertNotNull(someVar).toStmt())).toEqual('someVar!;');
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(
|
|
|
|
emitStmt(someVar.conditional(o.variable('trueCase'), o.variable('falseCase')).toStmt()))
|
2016-04-27 09:30:07 -04:00
|
|
|
.toEqual('(someVar? trueCase: falseCase);');
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
expect(emitStmt(lhs.equals(rhs).toStmt())).toEqual('(lhs == rhs);');
|
|
|
|
expect(emitStmt(lhs.notEquals(rhs).toStmt())).toEqual('(lhs != rhs);');
|
|
|
|
expect(emitStmt(lhs.identical(rhs).toStmt())).toEqual('(lhs === rhs);');
|
|
|
|
expect(emitStmt(lhs.notIdentical(rhs).toStmt())).toEqual('(lhs !== rhs);');
|
|
|
|
expect(emitStmt(lhs.minus(rhs).toStmt())).toEqual('(lhs - rhs);');
|
|
|
|
expect(emitStmt(lhs.plus(rhs).toStmt())).toEqual('(lhs + rhs);');
|
|
|
|
expect(emitStmt(lhs.divide(rhs).toStmt())).toEqual('(lhs / rhs);');
|
|
|
|
expect(emitStmt(lhs.multiply(rhs).toStmt())).toEqual('(lhs * rhs);');
|
|
|
|
expect(emitStmt(lhs.modulo(rhs).toStmt())).toEqual('(lhs % rhs);');
|
|
|
|
expect(emitStmt(lhs.and(rhs).toStmt())).toEqual('(lhs && rhs);');
|
|
|
|
expect(emitStmt(lhs.or(rhs).toStmt())).toEqual('(lhs || rhs);');
|
|
|
|
expect(emitStmt(lhs.lower(rhs).toStmt())).toEqual('(lhs < rhs);');
|
|
|
|
expect(emitStmt(lhs.lowerEquals(rhs).toStmt())).toEqual('(lhs <= rhs);');
|
|
|
|
expect(emitStmt(lhs.bigger(rhs).toStmt())).toEqual('(lhs > rhs);');
|
|
|
|
expect(emitStmt(lhs.biggerEquals(rhs).toStmt())).toEqual('(lhs >= rhs);');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support function expressions', () => {
|
2016-07-13 13:40:16 -04:00
|
|
|
expect(emitStmt(o.fn([], []).toStmt())).toEqual(['():void => {', '};'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(o.fn([], [new o.ReturnStatement(o.literal(1))], o.INT_TYPE).toStmt()))
|
|
|
|
.toEqual(['():number => {', ' return 1;\n};'].join('\n'));
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(o.fn([new o.FnParam('param1', o.INT_TYPE)], []).toStmt())).toEqual([
|
2016-07-13 13:40:16 -04:00
|
|
|
'(param1:number):void => {', '};'
|
2016-06-08 19:38:52 -04:00
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support function statements', () => {
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.DeclareFunctionStmt('someFn', [], []))).toEqual([
|
|
|
|
'function someFn():void {', '}'
|
|
|
|
].join('\n'));
|
2017-05-16 19:30:37 -04:00
|
|
|
expect(emitStmt(new o.DeclareFunctionStmt('someFn', [], [], null, [o.StmtModifier.Exported])))
|
|
|
|
.toEqual(['export function someFn():void {', '}'].join('\n'));
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.DeclareFunctionStmt(
|
|
|
|
'someFn', [], [new o.ReturnStatement(o.literal(1))], o.INT_TYPE)))
|
2016-01-06 17:13:44 -05:00
|
|
|
.toEqual(['function someFn():number {', ' return 1;', '}'].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(
|
|
|
|
emitStmt(new o.DeclareFunctionStmt('someFn', [new o.FnParam('param1', o.INT_TYPE)], [])))
|
|
|
|
.toEqual(['function someFn(param1:number):void {', '}'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
it('should support comments', () => {
|
|
|
|
expect(emitStmt(new o.CommentStmt('a\nb'))).toEqual(['// a', '// b'].join('\n'));
|
|
|
|
});
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
it('should support if stmt', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const trueCase = o.variable('trueCase').callFn([]).toStmt();
|
|
|
|
const falseCase = o.variable('falseCase').callFn([]).toStmt();
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.IfStmt(o.variable('cond'), [trueCase]))).toEqual([
|
|
|
|
'if (cond) { trueCase(); }'
|
|
|
|
].join('\n'));
|
|
|
|
expect(emitStmt(new o.IfStmt(o.variable('cond'), [trueCase], [falseCase]))).toEqual([
|
|
|
|
'if (cond) {', ' trueCase();', '} else {', ' falseCase();', '}'
|
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2019-12-06 03:41:44 -05:00
|
|
|
it('should support localized strings', () => {
|
2020-08-31 11:27:04 -04:00
|
|
|
const messageParts =
|
|
|
|
[new o.LiteralPiece('ab\\:c', {} as any), new o.LiteralPiece('d"e\'f', {} as any)];
|
|
|
|
const placeholders = [new o.PlaceholderPiece('ph1', {} as any)];
|
|
|
|
const expressions = [o.literal(7, o.NUMBER_TYPE).plus(o.literal(8, o.NUMBER_TYPE))];
|
|
|
|
const localizedString = o.localizedString({}, messageParts, placeholders, expressions);
|
|
|
|
expect(emitStmt(new o.ExpressionStatement(localizedString)))
|
2019-12-06 03:41:44 -05:00
|
|
|
.toEqual('$localize `ab\\\\:c${(7 + 8)}:ph1:d"e\'f`;');
|
|
|
|
});
|
|
|
|
|
2016-01-06 17:13:44 -05:00
|
|
|
it('should support try/catch', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const bodyStmt = o.variable('body').callFn([]).toStmt();
|
|
|
|
const catchStmt =
|
|
|
|
o.variable('catchFn').callFn([o.CATCH_ERROR_VAR, o.CATCH_STACK_VAR]).toStmt();
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.TryCatchStmt([bodyStmt], [catchStmt]))).toEqual([
|
|
|
|
'try {', ' body();', '} catch (error) {', ' const stack:any = error.stack;',
|
|
|
|
' catchFn(error,stack);', '}'
|
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
2020-04-08 13:14:18 -04:00
|
|
|
it('should support support throwing', () => {
|
|
|
|
expect(emitStmt(new o.ThrowStmt(someVar))).toEqual('throw someVar;');
|
|
|
|
});
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
describe('classes', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
let callSomeMethod: o.Statement;
|
2016-01-06 17:13:44 -05:00
|
|
|
|
2020-04-08 13:14:18 -04:00
|
|
|
beforeEach(() => {
|
|
|
|
callSomeMethod = o.THIS_EXPR.callMethod('someMethod', []).toStmt();
|
|
|
|
});
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
|
|
|
|
it('should support declaring classes', () => {
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt('SomeClass', null!, [], [], null!, []))).toEqual([
|
|
|
|
'class SomeClass {', '}'
|
|
|
|
].join('\n'));
|
|
|
|
expect(emitStmt(new o.ClassStmt('SomeClass', null!, [], [], null!, [], [
|
2017-05-16 19:30:37 -04:00
|
|
|
o.StmtModifier.Exported
|
|
|
|
]))).toEqual(['export class SomeClass {', '}'].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(
|
|
|
|
emitStmt(new o.ClassStmt('SomeClass', o.variable('SomeSuperClass'), [], [], null!, [])))
|
|
|
|
.toEqual(['class SomeClass extends SomeSuperClass {', '}'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support declaring constructors', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const superCall = o.SUPER_EXPR.callFn([o.variable('someParam')]).toStmt();
|
2017-03-24 12:59:58 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [], new o.ClassMethod(null!, [], []), [])))
|
2016-01-06 17:13:44 -05:00
|
|
|
.toEqual(['class SomeClass {', ' constructor() {', ' }', '}'].join('\n'));
|
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [],
|
|
|
|
new o.ClassMethod(null!, [new o.FnParam('someParam', o.INT_TYPE)], []), [])))
|
2016-01-06 17:13:44 -05:00
|
|
|
.toEqual(
|
|
|
|
['class SomeClass {', ' constructor(someParam:number) {', ' }', '}'].join('\n'));
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [], new o.ClassMethod(null!, [], [superCall]), [])))
|
2016-06-08 19:38:52 -04:00
|
|
|
.toEqual([
|
|
|
|
'class SomeClass {', ' constructor() {', ' super(someParam);', ' }', '}'
|
|
|
|
].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
|
|
|
'SomeClass', null!, [], [], new o.ClassMethod(null!, [], [callSomeMethod]), [])))
|
2016-06-08 19:38:52 -04:00
|
|
|
.toEqual([
|
|
|
|
'class SomeClass {', ' constructor() {', ' this.someMethod();', ' }', '}'
|
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support declaring fields', () => {
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [new o.ClassField('someField')], [], null!, [])))
|
2016-05-07 11:20:56 -04:00
|
|
|
.toEqual(['class SomeClass {', ' someField:any;', '}'].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
|
|
|
'SomeClass', null!, [new o.ClassField('someField', o.INT_TYPE)], [], null!, [])))
|
2016-01-06 17:13:44 -05:00
|
|
|
.toEqual(['class SomeClass {', ' someField:number;', '}'].join('\n'));
|
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!,
|
|
|
|
[new o.ClassField('someField', o.INT_TYPE, [o.StmtModifier.Private])], [], null!,
|
|
|
|
[])))
|
2016-08-22 17:28:09 -04:00
|
|
|
.toEqual(['class SomeClass {', ' /*private*/ someField:number;', '}'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support declaring getters', () => {
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [new o.ClassGetter('someGetter', [])], null!, [])))
|
2016-05-07 11:20:56 -04:00
|
|
|
.toEqual(['class SomeClass {', ' get someGetter():any {', ' }', '}'].join('\n'));
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [new o.ClassGetter('someGetter', [], o.INT_TYPE)], null!,
|
|
|
|
[])))
|
2016-01-06 17:13:44 -05:00
|
|
|
.toEqual(['class SomeClass {', ' get someGetter():number {', ' }', '}'].join('\n'));
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [new o.ClassGetter('someGetter', [callSomeMethod])],
|
|
|
|
null!, [])))
|
2016-06-08 19:38:52 -04:00
|
|
|
.toEqual([
|
|
|
|
'class SomeClass {', ' get someGetter():any {', ' this.someMethod();', ' }', '}'
|
|
|
|
].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(
|
|
|
|
emitStmt(new o.ClassStmt(
|
|
|
|
'SomeClass', null!, [],
|
|
|
|
[new o.ClassGetter('someGetter', [], null!, [o.StmtModifier.Private])], null!, [])))
|
2016-06-08 19:38:52 -04:00
|
|
|
.toEqual(
|
|
|
|
['class SomeClass {', ' private get someGetter():any {', ' }', '}'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support methods', () => {
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt('SomeClass', null!, [], [], null!, [
|
2016-06-08 19:38:52 -04:00
|
|
|
new o.ClassMethod('someMethod', [], [])
|
|
|
|
]))).toEqual(['class SomeClass {', ' someMethod():void {', ' }', '}'].join('\n'));
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(new o.ClassStmt('SomeClass', null!, [], [], null!, [
|
2016-06-08 19:38:52 -04:00
|
|
|
new o.ClassMethod('someMethod', [], [], o.INT_TYPE)
|
|
|
|
]))).toEqual(['class SomeClass {', ' someMethod():number {', ' }', '}'].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(
|
|
|
|
emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [], null!,
|
2016-01-06 17:13:44 -05:00
|
|
|
[new o.ClassMethod('someMethod', [new o.FnParam('someParam', o.INT_TYPE)], [])])))
|
2016-06-08 19:38:52 -04:00
|
|
|
.toEqual([
|
|
|
|
'class SomeClass {', ' someMethod(someParam:number):void {', ' }', '}'
|
|
|
|
].join('\n'));
|
|
|
|
expect(emitStmt(new o.ClassStmt(
|
2020-04-08 13:14:18 -04:00
|
|
|
'SomeClass', null!, [], [], null!,
|
2016-06-08 19:38:52 -04:00
|
|
|
[new o.ClassMethod('someMethod', [], [callSomeMethod])])))
|
|
|
|
.toEqual([
|
|
|
|
'class SomeClass {', ' someMethod():void {', ' this.someMethod();', ' }', '}'
|
|
|
|
].join('\n'));
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support builtin types', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const writeVarExpr = o.variable('a').set(o.NULL_EXPR);
|
2016-08-22 18:30:18 -04:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.DYNAMIC_TYPE)))
|
|
|
|
.toEqual('var a:any = (null as any);');
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.BOOL_TYPE)))
|
|
|
|
.toEqual('var a:boolean = (null as any);');
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.INT_TYPE)))
|
|
|
|
.toEqual('var a:number = (null as any);');
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.NUMBER_TYPE)))
|
|
|
|
.toEqual('var a:number = (null as any);');
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.STRING_TYPE)))
|
|
|
|
.toEqual('var a:string = (null as any);');
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.FUNCTION_TYPE)))
|
|
|
|
.toEqual('var a:Function = (null as any);');
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should support external types', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const writeVarExpr = o.variable('a').set(o.NULL_EXPR);
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.importType(sameModuleIdentifier))))
|
2016-08-22 18:30:18 -04:00
|
|
|
.toEqual('var a:someLocalId = (null as any);');
|
2016-06-08 19:38:52 -04:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(o.importType(externalModuleIdentifier)))).toEqual([
|
2017-05-11 13:28:48 -04:00
|
|
|
`import * as i0 from 'somePackage/someOtherPath';`,
|
|
|
|
`var a:i0.someExternalId = (null as any);`
|
2016-06-08 19:38:52 -04:00
|
|
|
].join('\n'));
|
2016-12-27 12:36:47 -05:00
|
|
|
});
|
|
|
|
|
2016-11-23 12:42:19 -05:00
|
|
|
it('should support expression types', () => {
|
2017-01-30 18:32:08 -05:00
|
|
|
expect(
|
|
|
|
emitStmt(o.variable('a').set(o.NULL_EXPR).toDeclStmt(o.expressionType(o.variable('b')))))
|
|
|
|
.toEqual('var a:b = (null as any);');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support expressions with type parameters', () => {
|
2016-11-23 12:42:19 -05:00
|
|
|
expect(emitStmt(o.variable('a')
|
|
|
|
.set(o.NULL_EXPR)
|
2017-01-30 18:32:08 -05:00
|
|
|
.toDeclStmt(o.importType(externalModuleIdentifier, [o.STRING_TYPE]))))
|
|
|
|
.toEqual([
|
2017-05-11 13:28:48 -04:00
|
|
|
`import * as i0 from 'somePackage/someOtherPath';`,
|
|
|
|
`var a:i0.someExternalId<string> = (null as any);`
|
2017-01-30 18:32:08 -05:00
|
|
|
].join('\n'));
|
2016-11-23 12:42:19 -05:00
|
|
|
});
|
|
|
|
|
2016-01-06 17:13:44 -05:00
|
|
|
it('should support combined types', () => {
|
2016-11-12 08:08:58 -05:00
|
|
|
const writeVarExpr = o.variable('a').set(o.NULL_EXPR);
|
2020-04-08 13:14:18 -04:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(new o.ArrayType(null!))))
|
2016-08-22 18:30:18 -04:00
|
|
|
.toEqual('var a:any[] = (null as any);');
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(new o.ArrayType(o.INT_TYPE))))
|
2016-08-22 18:30:18 -04:00
|
|
|
.toEqual('var a:number[] = (null as any);');
|
2016-01-06 17:13:44 -05:00
|
|
|
|
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(new o.MapType(null))))
|
2016-08-22 18:30:18 -04:00
|
|
|
.toEqual('var a:{[key: string]:any} = (null as any);');
|
2016-01-06 17:13:44 -05:00
|
|
|
expect(emitStmt(writeVarExpr.toDeclStmt(new o.MapType(o.INT_TYPE))))
|
2016-08-22 18:30:18 -04:00
|
|
|
.toEqual('var a:{[key: string]:number} = (null as any);');
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
2017-03-15 18:50:30 -04:00
|
|
|
|
2018-03-12 10:34:03 -04:00
|
|
|
describe('comments', () => {
|
|
|
|
it('should support a preamble', () => {
|
|
|
|
expect(emitStmt(o.variable('a').toStmt(), '/* SomePreamble */')).toBe([
|
|
|
|
'/* SomePreamble */', 'a;'
|
|
|
|
].join('\n'));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support singleline comments', () => {
|
|
|
|
expect(emitStmt(new o.CommentStmt('Simple comment'))).toBe('// Simple comment');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support multiline comments', () => {
|
|
|
|
expect(emitStmt(new o.CommentStmt('Multiline comment', true)))
|
|
|
|
.toBe('/* Multiline comment */');
|
|
|
|
expect(emitStmt(new o.CommentStmt(`Multiline\ncomment`, true)))
|
|
|
|
.toBe(`/* Multiline\ncomment */`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support JSDoc comments', () => {
|
|
|
|
expect(emitStmt(new o.JSDocCommentStmt([{text: 'Intro comment'}])))
|
|
|
|
.toBe(`/**\n * Intro comment\n */`);
|
|
|
|
expect(emitStmt(new o.JSDocCommentStmt([
|
|
|
|
{tagName: o.JSDocTagName.Desc, text: 'description'}
|
|
|
|
]))).toBe(`/**\n * @desc description\n */`);
|
|
|
|
expect(emitStmt(new o.JSDocCommentStmt([
|
|
|
|
{text: 'Intro comment'},
|
|
|
|
{tagName: o.JSDocTagName.Desc, text: 'description'},
|
|
|
|
{tagName: o.JSDocTagName.Id, text: '{number} identifier 123'},
|
|
|
|
])))
|
|
|
|
.toBe(
|
|
|
|
`/**\n * Intro comment\n * @desc description\n * @id {number} identifier 123\n */`);
|
|
|
|
});
|
2017-03-15 18:50:30 -04:00
|
|
|
});
|
2017-09-28 13:53:04 -04:00
|
|
|
|
|
|
|
describe('emitter context', () => {
|
|
|
|
it('should be able to back to the generating span', () => {
|
|
|
|
const file = new ParseSourceFile('some content', 'a.ts');
|
|
|
|
const returnSpan = new ParseSourceSpan(
|
|
|
|
new ParseLocation(file, 100, 10, 10), new ParseLocation(file, 200, 20, 10));
|
|
|
|
const referenceSpan = new ParseSourceSpan(
|
|
|
|
new ParseLocation(file, 150, 15, 10), new ParseLocation(file, 175, 17, 10));
|
|
|
|
const statements = [new o.ClassStmt(
|
|
|
|
'SomeClass', null, [], [], new o.ClassMethod(null, [], []),
|
|
|
|
[new o.ClassMethod('someMethod', [new o.FnParam('a', o.INT_TYPE)], [
|
|
|
|
o.variable('someVar', o.INT_TYPE).set(o.literal(0)).toDeclStmt(),
|
|
|
|
new o.ReturnStatement(o.variable('someVar', null, referenceSpan), returnSpan)
|
|
|
|
])])];
|
|
|
|
const {sourceText, context} =
|
2017-10-04 16:37:27 -04:00
|
|
|
emitter.emitStatementsAndContext('a.ts', statements, '/* some preamble /*\n\n');
|
2017-09-28 13:53:04 -04:00
|
|
|
const spanOf = (text: string, after: number = 0) => {
|
|
|
|
const location = sourceText.indexOf(text, after);
|
|
|
|
const {line, col} = calculateLineCol(sourceText, location);
|
|
|
|
return context.spanOf(line, col);
|
|
|
|
};
|
|
|
|
const returnLoc = sourceText.indexOf('return');
|
|
|
|
expect(spanOf('return someVar')).toEqual(returnSpan, 'return span calculated incorrectly');
|
|
|
|
expect(spanOf(';', returnLoc)).toEqual(returnSpan, 'reference span calculated incorrectly');
|
|
|
|
expect(spanOf('someVar', returnLoc))
|
|
|
|
.toEqual(referenceSpan, 'return span calculated incorrectly');
|
|
|
|
});
|
|
|
|
});
|
2016-01-06 17:13:44 -05:00
|
|
|
});
|
|
|
|
}
|
2017-09-28 13:53:04 -04:00
|
|
|
|
|
|
|
function calculateLineCol(text: string, offset: number): {line: number, col: number} {
|
|
|
|
const lines = text.split('\n');
|
|
|
|
let line = 0;
|
|
|
|
for (let cur = 0; cur < text.length; line++) {
|
|
|
|
const next = cur + lines[line].length + 1;
|
|
|
|
if (next > offset) {
|
|
|
|
return {line, col: offset - cur};
|
|
|
|
}
|
|
|
|
cur = next;
|
|
|
|
}
|
|
|
|
return {line, col: 0};
|
2018-03-12 10:34:03 -04:00
|
|
|
}
|