2016-03-22 20:11:42 -04:00
|
|
|
import * as ts from 'typescript';
|
2016-06-08 14:14:43 -04:00
|
|
|
|
2016-03-22 20:11:42 -04:00
|
|
|
import {MetadataCollector} from '../src/collector';
|
2016-05-31 14:00:39 -04:00
|
|
|
import {ClassMetadata, ConstructorMetadata, ModuleMetadata} from '../src/schema';
|
2016-03-22 20:11:42 -04:00
|
|
|
|
2016-06-08 14:14:43 -04:00
|
|
|
import {Directory, Host, expectValidSources} from './typescript.mocks';
|
2016-03-22 20:11:42 -04:00
|
|
|
|
|
|
|
describe('Collector', () => {
|
|
|
|
let host: ts.LanguageServiceHost;
|
|
|
|
let service: ts.LanguageService;
|
|
|
|
let program: ts.Program;
|
|
|
|
let collector: MetadataCollector;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
2016-05-31 14:00:39 -04:00
|
|
|
host = new Host(FILES, [
|
|
|
|
'/app/app.component.ts', '/app/cases-data.ts', '/app/error-cases.ts', '/promise.ts',
|
|
|
|
'/unsupported-1.ts', '/unsupported-2.ts'
|
|
|
|
]);
|
2016-03-22 20:11:42 -04:00
|
|
|
service = ts.createLanguageService(host);
|
|
|
|
program = service.getProgram();
|
2016-03-30 18:00:55 -04:00
|
|
|
collector = new MetadataCollector();
|
2016-03-22 20:11:42 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should not have errors in test data', () => { expectValidSources(service, program); });
|
|
|
|
|
|
|
|
it('should return undefined for modules that have no metadata', () => {
|
|
|
|
const sourceFile = program.getSourceFile('app/hero.ts');
|
2016-05-31 14:00:39 -04:00
|
|
|
const metadata = collector.getMetadata(sourceFile);
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(metadata).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
2016-05-26 13:45:37 -04:00
|
|
|
it('should be able to collect a simple component\'s metadata', () => {
|
2016-03-22 20:11:42 -04:00
|
|
|
const sourceFile = program.getSourceFile('app/hero-detail.component.ts');
|
2016-05-31 14:00:39 -04:00
|
|
|
const metadata = collector.getMetadata(sourceFile);
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(metadata).toEqual({
|
|
|
|
__symbolic: 'module',
|
2016-06-03 18:43:09 -04:00
|
|
|
version: 1,
|
2016-03-22 20:11:42 -04:00
|
|
|
metadata: {
|
|
|
|
HeroDetailComponent: {
|
|
|
|
__symbolic: 'class',
|
2016-05-26 13:45:37 -04:00
|
|
|
decorators: [{
|
|
|
|
__symbolic: 'call',
|
2016-05-31 14:00:39 -04:00
|
|
|
expression: {__symbolic: 'reference', module: 'angular2/core', name: 'Component'},
|
2016-05-26 13:45:37 -04:00
|
|
|
arguments: [{
|
|
|
|
selector: 'my-hero-detail',
|
|
|
|
template: `
|
2016-03-22 20:11:42 -04:00
|
|
|
<div *ngIf="hero">
|
|
|
|
<h2>{{hero.name}} details!</h2>
|
|
|
|
<div><label>id: </label>{{hero.id}}</div>
|
|
|
|
<div>
|
|
|
|
<label>name: </label>
|
|
|
|
<input [(ngModel)]="hero.name" placeholder="name"/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`
|
2016-05-26 13:45:37 -04:00
|
|
|
}]
|
|
|
|
}],
|
2016-03-22 20:11:42 -04:00
|
|
|
members: {
|
2016-05-26 13:45:37 -04:00
|
|
|
hero: [{
|
|
|
|
__symbolic: 'property',
|
|
|
|
decorators: [{
|
|
|
|
__symbolic: 'call',
|
|
|
|
expression:
|
2016-05-31 14:00:39 -04:00
|
|
|
{__symbolic: 'reference', module: 'angular2/core', name: 'Input'}
|
2016-05-26 13:45:37 -04:00
|
|
|
}]
|
|
|
|
}]
|
2016-03-22 20:11:42 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-05-26 13:45:37 -04:00
|
|
|
it('should be able to get a more complicated component\'s metadata', () => {
|
2016-03-22 20:11:42 -04:00
|
|
|
const sourceFile = program.getSourceFile('/app/app.component.ts');
|
2016-05-31 14:00:39 -04:00
|
|
|
const metadata = collector.getMetadata(sourceFile);
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(metadata).toEqual({
|
|
|
|
__symbolic: 'module',
|
2016-06-03 18:43:09 -04:00
|
|
|
version: 1,
|
2016-03-22 20:11:42 -04:00
|
|
|
metadata: {
|
|
|
|
AppComponent: {
|
|
|
|
__symbolic: 'class',
|
2016-05-26 13:45:37 -04:00
|
|
|
decorators: [{
|
|
|
|
__symbolic: 'call',
|
2016-05-31 14:00:39 -04:00
|
|
|
expression: {__symbolic: 'reference', module: 'angular2/core', name: 'Component'},
|
2016-05-26 13:45:37 -04:00
|
|
|
arguments: [{
|
|
|
|
selector: 'my-app',
|
|
|
|
template: `
|
2016-03-22 20:11:42 -04:00
|
|
|
<h2>My Heroes</h2>
|
|
|
|
<ul class="heroes">
|
|
|
|
<li *ngFor="#hero of heroes"
|
|
|
|
(click)="onSelect(hero)"
|
|
|
|
[class.selected]="hero === selectedHero">
|
|
|
|
<span class="badge">{{hero.id | lowercase}}</span> {{hero.name | uppercase}}
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
<my-hero-detail [hero]="selectedHero"></my-hero-detail>
|
|
|
|
`,
|
2016-05-26 13:45:37 -04:00
|
|
|
directives: [
|
|
|
|
{
|
|
|
|
__symbolic: 'reference',
|
2016-05-31 14:00:39 -04:00
|
|
|
module: './hero-detail.component',
|
2016-05-26 13:45:37 -04:00
|
|
|
name: 'HeroDetailComponent',
|
|
|
|
},
|
2016-05-31 14:00:39 -04:00
|
|
|
{__symbolic: 'reference', module: 'angular2/common', name: 'NgFor'}
|
2016-05-26 13:45:37 -04:00
|
|
|
],
|
2016-05-31 14:00:39 -04:00
|
|
|
providers: [{__symbolic: 'reference', module: './hero.service', default: true}],
|
2016-05-26 13:45:37 -04:00
|
|
|
pipes: [
|
2016-05-31 14:00:39 -04:00
|
|
|
{__symbolic: 'reference', module: 'angular2/common', name: 'LowerCasePipe'},
|
|
|
|
{__symbolic: 'reference', module: 'angular2/common', name: 'UpperCasePipe'}
|
2016-03-22 20:11:42 -04:00
|
|
|
]
|
2016-05-26 13:45:37 -04:00
|
|
|
}]
|
|
|
|
}],
|
2016-03-22 20:11:42 -04:00
|
|
|
members: {
|
2016-05-26 13:45:37 -04:00
|
|
|
__ctor__: [{
|
|
|
|
__symbolic: 'constructor',
|
2016-05-31 14:00:39 -04:00
|
|
|
parameters: [{__symbolic: 'reference', module: './hero.service', default: true}]
|
2016-05-26 13:45:37 -04:00
|
|
|
}],
|
2016-05-04 12:11:04 -04:00
|
|
|
onSelect: [{__symbolic: 'method'}],
|
|
|
|
ngOnInit: [{__symbolic: 'method'}],
|
|
|
|
getHeroes: [{__symbolic: 'method'}]
|
2016-03-22 20:11:42 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return the values of exported variables', () => {
|
|
|
|
const sourceFile = program.getSourceFile('/app/mock-heroes.ts');
|
2016-05-31 14:00:39 -04:00
|
|
|
const metadata = collector.getMetadata(sourceFile);
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(metadata).toEqual({
|
|
|
|
__symbolic: 'module',
|
2016-06-03 18:43:09 -04:00
|
|
|
version: 1,
|
2016-03-22 20:11:42 -04:00
|
|
|
metadata: {
|
|
|
|
HEROES: [
|
2016-05-26 13:45:37 -04:00
|
|
|
{'id': 11, 'name': 'Mr. Nice'}, {'id': 12, 'name': 'Narco'},
|
|
|
|
{'id': 13, 'name': 'Bombasto'}, {'id': 14, 'name': 'Celeritas'},
|
|
|
|
{'id': 15, 'name': 'Magneta'}, {'id': 16, 'name': 'RubberMan'},
|
|
|
|
{'id': 17, 'name': 'Dynama'}, {'id': 18, 'name': 'Dr IQ'}, {'id': 19, 'name': 'Magma'},
|
|
|
|
{'id': 20, 'name': 'Tornado'}
|
2016-03-22 20:11:42 -04:00
|
|
|
]
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-05-31 14:00:39 -04:00
|
|
|
it('should return undefined for modules that have no metadata', () => {
|
|
|
|
const sourceFile = program.getSourceFile('/app/error-cases.ts');
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(sourceFile).toBeTruthy(sourceFile);
|
2016-05-31 14:00:39 -04:00
|
|
|
const metadata = collector.getMetadata(sourceFile);
|
|
|
|
expect(metadata).toBeUndefined();
|
2016-03-22 20:11:42 -04:00
|
|
|
});
|
|
|
|
|
2016-01-22 13:51:16 -05:00
|
|
|
let casesFile: ts.SourceFile;
|
|
|
|
let casesMetadata: ModuleMetadata;
|
2016-03-25 13:16:06 -04:00
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
casesFile = program.getSourceFile('/app/cases-data.ts');
|
2016-05-31 14:00:39 -04:00
|
|
|
casesMetadata = collector.getMetadata(casesFile);
|
2016-03-25 13:16:06 -04:00
|
|
|
});
|
|
|
|
|
2016-05-31 14:00:39 -04:00
|
|
|
it('should provide any reference for an any ctor parameter type', () => {
|
2016-03-25 13:16:06 -04:00
|
|
|
const casesAny = <ClassMetadata>casesMetadata.metadata['CaseAny'];
|
2016-03-22 20:11:42 -04:00
|
|
|
expect(casesAny).toBeTruthy();
|
|
|
|
const ctorData = casesAny.members['__ctor__'];
|
2016-05-31 14:00:39 -04:00
|
|
|
expect(ctorData).toEqual(
|
|
|
|
[{__symbolic: 'constructor', parameters: [{__symbolic: 'reference', name: 'any'}]}]);
|
2016-03-22 20:11:42 -04:00
|
|
|
});
|
2016-03-25 13:16:06 -04:00
|
|
|
|
2016-04-26 00:29:06 -04:00
|
|
|
it('should record annotations on set and get declarations', () => {
|
2016-03-25 13:16:06 -04:00
|
|
|
const propertyData = {
|
2016-05-26 13:45:37 -04:00
|
|
|
name: [{
|
|
|
|
__symbolic: 'property',
|
|
|
|
decorators: [{
|
|
|
|
__symbolic: 'call',
|
|
|
|
expression: {__symbolic: 'reference', module: 'angular2/core', name: 'Input'},
|
|
|
|
arguments: ['firstName']
|
|
|
|
}]
|
|
|
|
}]
|
2016-03-25 13:16:06 -04:00
|
|
|
};
|
|
|
|
const caseGetProp = <ClassMetadata>casesMetadata.metadata['GetProp'];
|
|
|
|
expect(caseGetProp.members).toEqual(propertyData);
|
|
|
|
const caseSetProp = <ClassMetadata>casesMetadata.metadata['SetProp'];
|
|
|
|
expect(caseSetProp.members).toEqual(propertyData);
|
|
|
|
const caseFullProp = <ClassMetadata>casesMetadata.metadata['FullProp'];
|
|
|
|
expect(caseFullProp.members).toEqual(propertyData);
|
|
|
|
});
|
2016-05-31 14:00:39 -04:00
|
|
|
|
|
|
|
it('should record references to parameterized types', () => {
|
|
|
|
const casesForIn = <ClassMetadata>casesMetadata.metadata['NgFor'];
|
|
|
|
expect(casesForIn).toEqual({
|
|
|
|
__symbolic: 'class',
|
|
|
|
decorators: [{
|
|
|
|
__symbolic: 'call',
|
|
|
|
expression: {__symbolic: 'reference', module: 'angular2/core', name: 'Injectable'}
|
|
|
|
}],
|
|
|
|
members: {
|
|
|
|
__ctor__: [{
|
|
|
|
__symbolic: 'constructor',
|
|
|
|
parameters: [{
|
|
|
|
__symbolic: 'reference',
|
|
|
|
name: 'ClassReference',
|
|
|
|
arguments: [{__symbolic: 'reference', name: 'NgForRow'}]
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should report errors for destructured imports', () => {
|
|
|
|
let unsupported1 = program.getSourceFile('/unsupported-1.ts');
|
|
|
|
let metadata = collector.getMetadata(unsupported1);
|
|
|
|
expect(metadata).toEqual({
|
|
|
|
__symbolic: 'module',
|
2016-06-03 18:43:09 -04:00
|
|
|
version: 1,
|
2016-05-31 14:00:39 -04:00
|
|
|
metadata: {
|
|
|
|
a: {
|
2016-06-03 18:43:09 -04:00
|
|
|
__symbolic: 'error',
|
|
|
|
message: 'Destructuring declarations cannot be referenced statically',
|
|
|
|
line: 1,
|
|
|
|
character: 16
|
2016-05-31 14:00:39 -04:00
|
|
|
},
|
|
|
|
b: {
|
2016-06-03 18:43:09 -04:00
|
|
|
__symbolic: 'error',
|
|
|
|
message: 'Destructuring declarations cannot be referenced statically',
|
|
|
|
line: 1,
|
|
|
|
character: 18
|
2016-05-31 14:00:39 -04:00
|
|
|
},
|
|
|
|
c: {
|
2016-06-03 18:43:09 -04:00
|
|
|
__symbolic: 'error',
|
|
|
|
message: 'Destructuring declarations cannot be referenced statically',
|
|
|
|
line: 2,
|
|
|
|
character: 16
|
2016-05-31 14:00:39 -04:00
|
|
|
},
|
|
|
|
d: {
|
2016-06-03 18:43:09 -04:00
|
|
|
__symbolic: 'error',
|
|
|
|
message: 'Destructuring declarations cannot be referenced statically',
|
|
|
|
line: 2,
|
|
|
|
character: 18
|
2016-05-31 14:00:39 -04:00
|
|
|
},
|
|
|
|
e: {
|
|
|
|
__symbolic: 'error',
|
2016-06-03 18:43:09 -04:00
|
|
|
message: 'Only intialized variables and constants can be referenced statically',
|
|
|
|
line: 3,
|
|
|
|
character: 14
|
2016-05-31 14:00:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should report an error for refrences to unexpected types', () => {
|
|
|
|
let unsupported1 = program.getSourceFile('/unsupported-2.ts');
|
|
|
|
let metadata = collector.getMetadata(unsupported1);
|
|
|
|
let barClass = <ClassMetadata>metadata.metadata['Bar'];
|
|
|
|
let ctor = <ConstructorMetadata>barClass.members['__ctor__'][0];
|
|
|
|
let parameter = ctor.parameters[0];
|
2016-06-03 18:43:09 -04:00
|
|
|
expect(parameter).toEqual({
|
|
|
|
__symbolic: 'error',
|
|
|
|
message: 'Reference to non-exported class Foo',
|
|
|
|
line: 1,
|
|
|
|
character: 45
|
|
|
|
});
|
2016-05-31 14:00:39 -04:00
|
|
|
});
|
2016-03-22 20:11:42 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
// TODO: Do not use \` in a template literal as it confuses clang-format
|
|
|
|
const FILES: Directory = {
|
|
|
|
'app': {
|
|
|
|
'app.component.ts': `
|
2016-04-26 00:29:06 -04:00
|
|
|
import {Component as MyComponent, OnInit} from 'angular2/core';
|
|
|
|
import * as common from 'angular2/common';
|
2016-03-22 20:11:42 -04:00
|
|
|
import {Hero} from './hero';
|
|
|
|
import {HeroDetailComponent} from './hero-detail.component';
|
2016-04-26 00:29:06 -04:00
|
|
|
import HeroService from './hero.service';
|
|
|
|
// thrown away
|
|
|
|
import 'angular2/core';
|
2016-05-04 12:11:04 -04:00
|
|
|
|
2016-04-26 00:29:06 -04:00
|
|
|
@MyComponent({
|
2016-03-22 20:11:42 -04:00
|
|
|
selector: 'my-app',
|
2016-05-26 13:45:37 -04:00
|
|
|
template:` +
|
|
|
|
'`' +
|
|
|
|
`
|
2016-03-22 20:11:42 -04:00
|
|
|
<h2>My Heroes</h2>
|
|
|
|
<ul class="heroes">
|
|
|
|
<li *ngFor="#hero of heroes"
|
|
|
|
(click)="onSelect(hero)"
|
|
|
|
[class.selected]="hero === selectedHero">
|
|
|
|
<span class="badge">{{hero.id | lowercase}}</span> {{hero.name | uppercase}}
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
<my-hero-detail [hero]="selectedHero"></my-hero-detail>
|
|
|
|
` +
|
2016-05-26 13:45:37 -04:00
|
|
|
'`' +
|
|
|
|
`,
|
2016-04-26 00:29:06 -04:00
|
|
|
directives: [HeroDetailComponent, common.NgFor],
|
2016-03-22 20:11:42 -04:00
|
|
|
providers: [HeroService],
|
2016-04-26 00:29:06 -04:00
|
|
|
pipes: [common.LowerCasePipe, common.UpperCasePipe]
|
2016-03-22 20:11:42 -04:00
|
|
|
})
|
|
|
|
export class AppComponent implements OnInit {
|
|
|
|
public title = 'Tour of Heroes';
|
|
|
|
public heroes: Hero[];
|
|
|
|
public selectedHero: Hero;
|
|
|
|
|
|
|
|
constructor(private _heroService: HeroService) { }
|
|
|
|
|
|
|
|
onSelect(hero: Hero) { this.selectedHero = hero; }
|
|
|
|
|
|
|
|
ngOnInit() {
|
|
|
|
this.getHeroes()
|
|
|
|
}
|
|
|
|
|
|
|
|
getHeroes() {
|
|
|
|
this._heroService.getHeroesSlowly().then(heros => this.heroes = heros);
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
'hero.ts': `
|
|
|
|
export interface Hero {
|
|
|
|
id: number;
|
|
|
|
name: string;
|
|
|
|
}`,
|
|
|
|
'hero-detail.component.ts': `
|
|
|
|
import {Component, Input} from 'angular2/core';
|
|
|
|
import {Hero} from './hero';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: 'my-hero-detail',
|
2016-05-26 13:45:37 -04:00
|
|
|
template: ` +
|
|
|
|
'`' +
|
|
|
|
`
|
2016-03-22 20:11:42 -04:00
|
|
|
<div *ngIf="hero">
|
|
|
|
<h2>{{hero.name}} details!</h2>
|
|
|
|
<div><label>id: </label>{{hero.id}}</div>
|
|
|
|
<div>
|
|
|
|
<label>name: </label>
|
|
|
|
<input [(ngModel)]="hero.name" placeholder="name"/>
|
|
|
|
</div>
|
|
|
|
</div>
|
2016-05-26 13:45:37 -04:00
|
|
|
` +
|
|
|
|
'`' +
|
|
|
|
`,
|
2016-03-22 20:11:42 -04:00
|
|
|
})
|
|
|
|
export class HeroDetailComponent {
|
|
|
|
@Input() public hero: Hero;
|
|
|
|
}`,
|
|
|
|
'mock-heroes.ts': `
|
2016-04-26 00:29:06 -04:00
|
|
|
import {Hero as Hero} from './hero';
|
2016-03-22 20:11:42 -04:00
|
|
|
|
|
|
|
export const HEROES: Hero[] = [
|
|
|
|
{"id": 11, "name": "Mr. Nice"},
|
|
|
|
{"id": 12, "name": "Narco"},
|
|
|
|
{"id": 13, "name": "Bombasto"},
|
|
|
|
{"id": 14, "name": "Celeritas"},
|
|
|
|
{"id": 15, "name": "Magneta"},
|
|
|
|
{"id": 16, "name": "RubberMan"},
|
|
|
|
{"id": 17, "name": "Dynama"},
|
|
|
|
{"id": 18, "name": "Dr IQ"},
|
|
|
|
{"id": 19, "name": "Magma"},
|
|
|
|
{"id": 20, "name": "Tornado"}
|
|
|
|
];`,
|
2016-04-26 00:29:06 -04:00
|
|
|
'default-exporter.ts': `
|
|
|
|
let a: string;
|
|
|
|
export default a;
|
|
|
|
`,
|
2016-03-22 20:11:42 -04:00
|
|
|
'hero.service.ts': `
|
|
|
|
import {Injectable} from 'angular2/core';
|
|
|
|
import {HEROES} from './mock-heroes';
|
|
|
|
import {Hero} from './hero';
|
|
|
|
|
|
|
|
@Injectable()
|
2016-04-26 00:29:06 -04:00
|
|
|
class HeroService {
|
2016-03-22 20:11:42 -04:00
|
|
|
getHeros() {
|
|
|
|
return Promise.resolve(HEROES);
|
|
|
|
}
|
|
|
|
|
|
|
|
getHeroesSlowly() {
|
|
|
|
return new Promise<Hero[]>(resolve =>
|
|
|
|
setTimeout(()=>resolve(HEROES), 2000)); // 2 seconds
|
|
|
|
}
|
2016-04-26 00:29:06 -04:00
|
|
|
}
|
|
|
|
export default HeroService;`,
|
2016-03-22 20:11:42 -04:00
|
|
|
'cases-data.ts': `
|
2016-03-25 13:16:06 -04:00
|
|
|
import {Injectable, Input} from 'angular2/core';
|
2016-03-22 20:11:42 -04:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class CaseAny {
|
|
|
|
constructor(param: any) {}
|
|
|
|
}
|
2016-05-04 12:11:04 -04:00
|
|
|
|
2016-03-25 13:16:06 -04:00
|
|
|
@Injectable()
|
|
|
|
export class GetProp {
|
|
|
|
private _name: string;
|
|
|
|
@Input('firstName') get name(): string {
|
|
|
|
return this._name;
|
|
|
|
}
|
|
|
|
}
|
2016-05-04 12:11:04 -04:00
|
|
|
|
2016-03-25 13:16:06 -04:00
|
|
|
@Injectable()
|
|
|
|
export class SetProp {
|
|
|
|
private _name: string;
|
|
|
|
@Input('firstName') set name(value: string) {
|
|
|
|
this._name = value;
|
|
|
|
}
|
|
|
|
}
|
2016-05-04 12:11:04 -04:00
|
|
|
|
2016-03-25 13:16:06 -04:00
|
|
|
@Injectable()
|
|
|
|
export class FullProp {
|
|
|
|
private _name: string;
|
|
|
|
@Input('firstName') get name(): string {
|
|
|
|
return this._name;
|
|
|
|
}
|
|
|
|
set name(value: string) {
|
|
|
|
this._name = value;
|
|
|
|
}
|
|
|
|
}
|
2016-05-31 14:00:39 -04:00
|
|
|
|
|
|
|
export class ClassReference<T> { }
|
|
|
|
export class NgForRow {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class NgFor {
|
|
|
|
constructor (public ref: ClassReference<NgForRow>) {}
|
|
|
|
}
|
2016-03-22 20:11:42 -04:00
|
|
|
`,
|
2016-05-31 14:00:39 -04:00
|
|
|
'error-cases.ts': `
|
2016-04-26 00:29:06 -04:00
|
|
|
import HeroService from './hero.service';
|
2016-03-22 20:11:42 -04:00
|
|
|
|
|
|
|
export class CaseCtor {
|
|
|
|
constructor(private _heroService: HeroService) { }
|
|
|
|
}
|
|
|
|
`
|
|
|
|
},
|
|
|
|
'promise.ts': `
|
|
|
|
interface PromiseLike<T> {
|
|
|
|
then<TResult>(onfulfilled?: (value: T) => TResult | PromiseLike<TResult>, onrejected?: (reason: any) => TResult | PromiseLike<TResult>): PromiseLike<TResult>;
|
|
|
|
then<TResult>(onfulfilled?: (value: T) => TResult | PromiseLike<TResult>, onrejected?: (reason: any) => void): PromiseLike<TResult>;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface Promise<T> {
|
|
|
|
then<TResult>(onfulfilled?: (value: T) => TResult | PromiseLike<TResult>, onrejected?: (reason: any) => TResult | PromiseLike<TResult>): Promise<TResult>;
|
|
|
|
then<TResult>(onfulfilled?: (value: T) => TResult | PromiseLike<TResult>, onrejected?: (reason: any) => void): Promise<TResult>;
|
|
|
|
catch(onrejected?: (reason: any) => T | PromiseLike<T>): Promise<T>;
|
|
|
|
catch(onrejected?: (reason: any) => void): Promise<T>;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface PromiseConstructor {
|
|
|
|
prototype: Promise<any>;
|
|
|
|
new <T>(executor: (resolve: (value?: T | PromiseLike<T>) => void, reject: (reason?: any) => void) => void): Promise<T>;
|
|
|
|
reject(reason: any): Promise<void>;
|
|
|
|
reject<T>(reason: any): Promise<T>;
|
|
|
|
resolve<T>(value: T | PromiseLike<T>): Promise<T>;
|
|
|
|
resolve(): Promise<void>;
|
|
|
|
}
|
|
|
|
|
|
|
|
declare var Promise: PromiseConstructor;
|
|
|
|
`,
|
2016-05-31 14:00:39 -04:00
|
|
|
'unsupported-1.ts': `
|
|
|
|
export let {a, b} = {a: 1, b: 2};
|
|
|
|
export let [c, d] = [1, 2];
|
|
|
|
export let e;
|
|
|
|
`,
|
|
|
|
'unsupported-2.ts': `
|
|
|
|
import {Injectable} from 'angular2/core';
|
|
|
|
|
|
|
|
class Foo {}
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class Bar {
|
|
|
|
constructor(private f: Foo) {}
|
|
|
|
}
|
|
|
|
`,
|
2016-03-22 20:11:42 -04:00
|
|
|
'node_modules': {
|
|
|
|
'angular2': {
|
|
|
|
'core.d.ts': `
|
|
|
|
export interface Type extends Function { }
|
|
|
|
export interface TypeDecorator {
|
|
|
|
<T extends Type>(type: T): T;
|
|
|
|
(target: Object, propertyKey?: string | symbol, parameterIndex?: number): void;
|
|
|
|
annotations: any[];
|
|
|
|
}
|
|
|
|
export interface ComponentDecorator extends TypeDecorator { }
|
|
|
|
export interface ComponentFactory {
|
|
|
|
(obj: {
|
|
|
|
selector?: string;
|
|
|
|
inputs?: string[];
|
|
|
|
outputs?: string[];
|
|
|
|
properties?: string[];
|
|
|
|
events?: string[];
|
|
|
|
host?: {
|
|
|
|
[key: string]: string;
|
|
|
|
};
|
|
|
|
bindings?: any[];
|
|
|
|
providers?: any[];
|
|
|
|
exportAs?: string;
|
|
|
|
moduleId?: string;
|
|
|
|
queries?: {
|
|
|
|
[key: string]: any;
|
|
|
|
};
|
|
|
|
viewBindings?: any[];
|
|
|
|
viewProviders?: any[];
|
|
|
|
templateUrl?: string;
|
|
|
|
template?: string;
|
|
|
|
styleUrls?: string[];
|
|
|
|
styles?: string[];
|
|
|
|
directives?: Array<Type | any[]>;
|
|
|
|
pipes?: Array<Type | any[]>;
|
|
|
|
}): ComponentDecorator;
|
|
|
|
}
|
|
|
|
export declare var Component: ComponentFactory;
|
|
|
|
export interface InputFactory {
|
|
|
|
(bindingPropertyName?: string): any;
|
|
|
|
new (bindingPropertyName?: string): any;
|
|
|
|
}
|
|
|
|
export declare var Input: InputFactory;
|
|
|
|
export interface InjectableFactory {
|
|
|
|
(): any;
|
|
|
|
}
|
|
|
|
export declare var Injectable: InjectableFactory;
|
|
|
|
export interface OnInit {
|
|
|
|
ngOnInit(): any;
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
'common.d.ts': `
|
|
|
|
export declare class NgFor {
|
|
|
|
ngForOf: any;
|
|
|
|
ngForTemplate: any;
|
|
|
|
ngDoCheck(): void;
|
|
|
|
}
|
|
|
|
export declare class LowerCasePipe {
|
|
|
|
transform(value: string, args?: any[]): string;
|
|
|
|
}
|
|
|
|
export declare class UpperCasePipe {
|
|
|
|
transform(value: string, args?: any[]): string;
|
|
|
|
}
|
|
|
|
`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|