feat(browser): use AppModules for bootstrap in the browser

This introduces the `BrowserModule` to be used for long form
bootstrap and offline compile bootstrap:

```
@AppModule({
  modules: [BrowserModule],
  precompile: [MainComponent],
  providers: […], // additional providers
  directives: […], // additional platform directives
  pipes: […] // additional platform pipes
})
class MyModule {
  constructor(appRef: ApplicationRef) {
    appRef.bootstrap(MainComponent);
  }
}

// offline compile
import {bootstrapModuleFactory} from ‘@angular/platform-browser’;
bootstrapModuleFactory(MyModuleNgFactory);

// runtime compile long form
import {bootstrapModule} from ‘@angular/platform-browser-dynamic’;
bootstrapModule(MyModule);
```

The short form, `bootstrap(...)`, can now creates a module on the fly,
given `directives`, `pipes, `providers`, `precompile` and `modules`
properties.

Related changes:
- make `SanitizationService`, `SecurityContext` public in `@angular/core` so that the offline compiler can resolve the token
- move `AnimationDriver` to `platform-browser` and make it
  public so that the offline compiler can resolve the token

BREAKING CHANGES:
- short form bootstrap does no longer allow
  to inject compiler internals (i.e. everything 
  from `@angular/compiler). Inject `Compiler` instead.
  To provide custom providers for the compiler,
  create a custom compiler via `browserCompiler({providers: [...]})`
  and pass that into the `bootstrap` method.
This commit is contained in:
Tobias Bosch 2016-06-30 13:07:17 -07:00
parent 74b45dfbf8
commit 3f55aa609f
71 changed files with 793 additions and 406 deletions

View File

@ -27,13 +27,30 @@ Then you can add an import statement in the `bootstrap` allowing you to bootstra
generated code: generated code:
```typescript ```typescript
import {ComponentResolver, ReflectiveInjector, coreBootstrap} from '@angular/core'; main_module.ts
import {BROWSER_APP_PROVIDERS, browserPlatform} from '@angular/platform-browser'; -------------
import {BrowserModule} from '@angular/platform-browser';
import {Component, AppModule, ApplicationRef} from '@angular/core';
import {MyComponentNgFactory} from './mycomponent.ngfactory'; @Component(...)
export class MyComponent {}
const appInjector = ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, browserPlatform().injector); @AppModule({
coreBootstrap(MyComponentNgFactory, appInjector); modules: [BrowserModule],
precompile: [MyComponent]
})
export class MainModule {
constructor(appRef: ApplicationRef) {
appRef.bootstrap(MyComponent);
}
}
bootstrap.ts
-------------
import {MainModuleNgFactory} from './main_module.ngfactory';
MainModuleNgFactory.create(browserPlatform().injector);
``` ```
## Configuration ## Configuration

View File

@ -12,7 +12,7 @@ import {Component} from '@angular/core';
selector: 'my-comp', selector: 'my-comp',
template: '<div></div>', template: '<div></div>',
}) })
export class MyComp { export class MultipleComponentsMyComp {
} }
@Component({ @Component({

View File

@ -9,16 +9,16 @@
import {FORM_DIRECTIVES, NgFor, NgIf} from '@angular/common'; import {FORM_DIRECTIVES, NgFor, NgIf} from '@angular/common';
import {Component, Inject} from '@angular/core'; import {Component, Inject} from '@angular/core';
import {MyComp} from './a/multiple_components'; import {MultipleComponentsMyComp} from './a/multiple_components';
@Component({ @Component({
selector: 'basic', selector: 'basic',
templateUrl: './basic.html', templateUrl: './basic.html',
styles: ['.red { color: red }'], styles: ['.red { color: red }'],
styleUrls: ['./basic.css'], styleUrls: ['./basic.css'],
directives: [MyComp, FORM_DIRECTIVES, NgIf, NgFor] directives: [MultipleComponentsMyComp, FORM_DIRECTIVES, NgIf, NgFor]
}) })
export class Basic { export class BasicComp {
ctxProp: string; ctxProp: string;
ctxBool: boolean; ctxBool: boolean;
ctxArr: any[] = []; ctxArr: any[] = [];

View File

@ -6,12 +6,8 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {ReflectiveInjector, coreBootstrap} from '@angular/core'; import {browserPlatform} from '@angular/platform-browser';
import {BROWSER_APP_PROVIDERS, browserPlatform} from '@angular/platform-browser'; import {BasicComp} from './basic';
import {MainModuleNgFactory} from './module.ngfactory';
import {Basic} from './basic'; MainModuleNgFactory.create().instance.appRef.bootstrap(BasicComp);
import {BasicNgFactory} from './basic.ngfactory';
const appInjector =
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, browserPlatform().injector);
coreBootstrap(BasicNgFactory, appInjector);

View File

@ -0,0 +1,24 @@
/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* 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
*/
import {AppModule, ApplicationRef} from '@angular/core';
import {BrowserModule} from '@angular/platform-browser';
import {AnimateCmp} from './animate';
import {BasicComp} from './basic';
import {CompWithPrecompile} from './precompile';
import {ProjectingComp} from './projection';
import {CompWithChildQuery} from './queries';
@AppModule({
modules: [BrowserModule],
precompile: [AnimateCmp, BasicComp, CompWithPrecompile, ProjectingComp, CompWithChildQuery]
})
export class MainModule {
constructor(public appRef: ApplicationRef) {}
}

View File

@ -8,6 +8,7 @@
import {LowerCasePipe, NgIf} from '@angular/common'; import {LowerCasePipe, NgIf} from '@angular/common';
import {AppModule, Component, ComponentFactoryResolver, Injectable} from '@angular/core'; import {AppModule, Component, ComponentFactoryResolver, Injectable} from '@angular/core';
import {BrowserModule} from '@angular/platform-browser';
@Injectable() @Injectable()
export class SomeService { export class SomeService {
@ -39,7 +40,7 @@ export class NestedModule {
pipes: [LowerCasePipe], pipes: [LowerCasePipe],
providers: [SomeService], providers: [SomeService],
precompile: [SomeComp], precompile: [SomeComp],
modules: [NestedModule] modules: [NestedModule, BrowserModule]
}) })
export class SomeModule { export class SomeModule {
} }
@ -48,6 +49,7 @@ export class SomeModule {
directives: [NgIf], directives: [NgIf],
pipes: [LowerCasePipe], pipes: [LowerCasePipe],
precompile: [ParentComp], precompile: [ParentComp],
modules: [BrowserModule]
}) })
export class SomeModuleUsingParentComp { export class SomeModuleUsingParentComp {
} }

View File

@ -7,12 +7,9 @@
*/ */
import {Component, ComponentFactoryResolver, Inject, OpaqueToken} from '@angular/core'; import {Component, ComponentFactoryResolver, Inject, OpaqueToken} from '@angular/core';
import {BasicComp} from './basic';
@Component({selector: 'cmp', template: ''}) @Component({selector: 'cmp-precompile', template: '', precompile: [BasicComp]})
export class SomeComp {
}
@Component({selector: 'cmp-precompile', template: '', precompile: [SomeComp]})
export class CompWithPrecompile { export class CompWithPrecompile {
constructor(public cfr: ComponentFactoryResolver) {} constructor(public cfr: ComponentFactoryResolver) {}
} }

View File

@ -9,13 +9,13 @@
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({selector: 'comp-with-proj', template: '<ng-content></ng-content>'}) @Component({selector: 'comp-with-proj', template: '<ng-content></ng-content>'})
export class CompWithProjection { export class CompWithNgContent {
} }
@Component({ @Component({
selector: 'main', selector: 'main',
template: '<comp-with-proj><span greeting="Hello world!"></span></comp-with-proj>', template: '<comp-with-proj><span greeting="Hello world!"></span></comp-with-proj>',
directives: [CompWithProjection] directives: [CompWithNgContent]
}) })
export class MainComp { export class ProjectingComp {
} }

View File

@ -5,19 +5,10 @@
* Use of this source code is governed by an MIT-style license that can be * 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 * found in the LICENSE file at https://angular.io/license
*/ */
import './init';
require('reflect-metadata'); import {DebugElement} from '@angular/core';
require('zone.js/dist/zone-node.js'); import {AnimateCmp} from '../src/animate';
require('zone.js/dist/long-stack-trace-zone.js'); import {createComponent} from './util';
import {AnimateCmpNgFactory} from '../src/animate.ngfactory';
import {ReflectiveInjector, DebugElement, getDebugNode, lockRunMode} from '@angular/core';
import {serverPlatform} from '@angular/platform-server';
import {BROWSER_APP_PROVIDERS} from '@angular/platform-browser';
// Need to lock the mode explicitely as this test is not using Angular's testing framework.
lockRunMode();
describe('template codegen output', () => { describe('template codegen output', () => {
function findTargetElement(elm: DebugElement): DebugElement { function findTargetElement(elm: DebugElement): DebugElement {
@ -27,23 +18,21 @@ describe('template codegen output', () => {
} }
it('should apply the animate states to the element', (done) => { it('should apply the animate states to the element', (done) => {
const appInjector = const compFixture = createComponent(AnimateCmp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var comp = AnimateCmpNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
var targetDebugElement = findTargetElement(<DebugElement>debugElement); var targetDebugElement = findTargetElement(<DebugElement>debugElement);
comp.instance.setAsOpen(); compFixture.componentInstance.setAsOpen();
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
setTimeout(() => { setTimeout(() => {
expect(targetDebugElement.styles['height']).toEqual(null); expect(targetDebugElement.styles['height']).toEqual(null);
expect(targetDebugElement.styles['borderColor']).toEqual('green'); expect(targetDebugElement.styles['borderColor']).toEqual('green');
expect(targetDebugElement.styles['color']).toEqual('green'); expect(targetDebugElement.styles['color']).toEqual('green');
comp.instance.setAsClosed(); compFixture.componentInstance.setAsClosed();
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
setTimeout(() => { setTimeout(() => {
expect(targetDebugElement.styles['height']).toEqual('0px'); expect(targetDebugElement.styles['height']).toEqual('0px');
@ -55,23 +44,21 @@ describe('template codegen output', () => {
}); });
it('should apply the default animate state to the element', (done) => { it('should apply the default animate state to the element', (done) => {
const appInjector = const compFixture = createComponent(AnimateCmp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var comp = AnimateCmpNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
var targetDebugElement = findTargetElement(<DebugElement>debugElement); var targetDebugElement = findTargetElement(<DebugElement>debugElement);
comp.instance.setAsSomethingElse(); compFixture.componentInstance.setAsSomethingElse();
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
setTimeout(() => { setTimeout(() => {
expect(targetDebugElement.styles['height']).toEqual(null); expect(targetDebugElement.styles['height']).toEqual(null);
expect(targetDebugElement.styles['borderColor']).toEqual('black'); expect(targetDebugElement.styles['borderColor']).toEqual('black');
expect(targetDebugElement.styles['color']).toEqual('black'); expect(targetDebugElement.styles['color']).toEqual('black');
comp.instance.setAsClosed(); compFixture.componentInstance.setAsClosed();
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
setTimeout(() => { setTimeout(() => {
expect(targetDebugElement.styles['height']).not.toEqual(null); expect(targetDebugElement.styles['height']).not.toEqual(null);

View File

@ -5,68 +5,52 @@
* Use of this source code is governed by an MIT-style license that can be * 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 * found in the LICENSE file at https://angular.io/license
*/ */
import './init';
import {ComponentFactoryResolver, DebugElement, ReflectiveInjector, getDebugNode, lockRunMode} from '@angular/core'; import {NestedModule, NestedService, ParentComp, SomeComp, SomeModule, SomeService} from '../src/module_fixtures';
import {BROWSER_APP_PROVIDERS, By} from '@angular/platform-browser'; import {SomeModuleNgFactory, SomeModuleUsingParentCompNgFactory} from '../src/module_fixtures.ngfactory';
import {serverPlatform} from '@angular/platform-server'; import {createComponent, createModule} from './util';
import {NestedModule, NestedService, ParentComp, SomeComp, SomeModule, SomeService} from '../src/app_module';
import {SomeModuleNgFactory, SomeModuleUsingParentCompNgFactory} from '../src/app_module.ngfactory';
// Need to lock the mode explicitely as this test is not using Angular's testing framework.
lockRunMode();
describe('AppModule', () => { describe('AppModule', () => {
it('should support providers', () => { it('should support providers', () => {
var moduleRef = SomeModuleNgFactory.create(); var moduleRef = createModule(SomeModuleNgFactory);
expect(moduleRef.instance instanceof SomeModule).toBe(true); expect(moduleRef.instance instanceof SomeModule).toBe(true);
expect(moduleRef.injector.get(SomeModule) instanceof SomeModule).toBe(true); expect(moduleRef.injector.get(SomeModule) instanceof SomeModule).toBe(true);
expect(moduleRef.injector.get(SomeService) instanceof SomeService).toBe(true); expect(moduleRef.injector.get(SomeService) instanceof SomeService).toBe(true);
}); });
it('should support precompile components', () => { it('should support precompile components', () => {
const appInjector = var moduleRef = createModule(SomeModuleNgFactory);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var cf = moduleRef.componentFactoryResolver.resolveComponentFactory(SomeComp);
var moduleRef = SomeModuleNgFactory.create(appInjector);
var cf = moduleRef.injector.get(ComponentFactoryResolver).resolveComponentFactory(SomeComp);
expect(cf.componentType).toBe(SomeComp); expect(cf.componentType).toBe(SomeComp);
var comp = cf.create(moduleRef.injector); var compRef = cf.create(moduleRef.injector);
expect(compRef.instance instanceof SomeComp).toBe(true);
}); });
it('should support module directives and pipes', () => { it('should support module directives and pipes', () => {
const appInjector = var compFixture = createComponent(SomeComp, SomeModuleNgFactory);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var moduleRef = SomeModuleNgFactory.create(appInjector);
var cf = moduleRef.injector.get(ComponentFactoryResolver).resolveComponentFactory(SomeComp);
var comp = cf.create(moduleRef.injector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
// NgIf should work, is being used as module directive // NgIf should work, is being used as module directive
expect(debugElement.children.length).toBe(1); expect(debugElement.children.length).toBe(1);
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
expect(debugElement.children.length).toBe(2); expect(debugElement.children.length).toBe(2);
expect(debugElement.children[0].properties['title']).toBe('hello'); expect(debugElement.children[0].properties['title']).toBe('hello');
}); });
it('should support module directives and pipes on nested components', () => { it('should support module directives and pipes on nested components', () => {
const appInjector = var compFixture = createComponent(ParentComp, SomeModuleUsingParentCompNgFactory);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var moduleRef = SomeModuleUsingParentCompNgFactory.create(appInjector);
var cf = moduleRef.injector.get(ComponentFactoryResolver).resolveComponentFactory(ParentComp);
var comp = cf.create(moduleRef.injector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
debugElement = debugElement.children[0]; debugElement = debugElement.children[0];
// NgIf should work, is being used as module directive // NgIf should work, is being used as module directive
expect(debugElement.children.length).toBe(1); expect(debugElement.children.length).toBe(1);
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
expect(debugElement.children.length).toBe(2); expect(debugElement.children.length).toBe(2);
expect(debugElement.children[0].properties['title']).toBe('hello'); expect(debugElement.children[0].properties['title']).toBe('hello');
}); });
it('should support child moduless', () => { it('should support child moduless', () => {
var moduleRef = SomeModuleNgFactory.create(); var moduleRef = createModule(SomeModuleNgFactory);
expect(moduleRef.instance instanceof SomeModule).toBe(true); expect(moduleRef.instance instanceof SomeModule).toBe(true);
expect(moduleRef.injector.get(NestedModule) instanceof NestedModule).toBe(true); expect(moduleRef.injector.get(NestedModule) instanceof NestedModule).toBe(true);
expect(moduleRef.injector.get(NestedService) instanceof NestedService).toBe(true); expect(moduleRef.injector.get(NestedService) instanceof NestedService).toBe(true);

View File

@ -5,24 +5,13 @@
* Use of this source code is governed by an MIT-style license that can be * 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 * found in the LICENSE file at https://angular.io/license
*/ */
import './init';
// Only needed to satisfy the check in core/src/util/decorators.ts
// TODO(alexeagle): maybe remove that check?
require('reflect-metadata');
require('zone.js/dist/zone-node.js');
require('zone.js/dist/long-stack-trace-zone.js');
import * as fs from 'fs'; import * as fs from 'fs';
import * as path from 'path'; import * as path from 'path';
import {BasicNgFactory} from '../src/basic.ngfactory';
import {MyComp} from '../src/a/multiple_components';
import {ReflectiveInjector, DebugElement, getDebugNode, lockRunMode} from '@angular/core';
import {BROWSER_APP_PROVIDERS} from '@angular/platform-browser';
import {serverPlatform} from '@angular/platform-server';
// Need to lock the mode explicitely as this test is not using Angular's testing framework. import {MultipleComponentsMyComp} from '../src/a/multiple_components';
lockRunMode(); import {BasicComp} from '../src/basic';
import {createComponent} from './util';
describe('template codegen output', () => { describe('template codegen output', () => {
const outDir = 'src'; const outDir = 'src';
@ -48,35 +37,29 @@ describe('template codegen output', () => {
}); });
it('should be able to create the basic component', () => { it('should be able to create the basic component', () => {
const appInjector = var compFixture = createComponent(BasicComp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); expect(compFixture.componentInstance).toBeTruthy();
var comp = BasicNgFactory.create(appInjector);
expect(comp.instance).toBeTruthy();
}); });
it('should support ngIf', () => { it('should support ngIf', () => {
const appInjector = var compFixture = createComponent(BasicComp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var comp = BasicNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
expect(debugElement.children.length).toBe(2); expect(debugElement.children.length).toBe(2);
comp.instance.ctxBool = true; compFixture.componentInstance.ctxBool = true;
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
expect(debugElement.children.length).toBe(3); expect(debugElement.children.length).toBe(3);
expect(debugElement.children[2].injector.get(MyComp)).toBeTruthy(); expect(debugElement.children[2].injector.get(MultipleComponentsMyComp)).toBeTruthy();
}); });
it('should support ngFor', () => { it('should support ngFor', () => {
const appInjector = var compFixture = createComponent(BasicComp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = compFixture.debugElement;
var comp = BasicNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(comp.location.nativeElement);
expect(debugElement.children.length).toBe(2); expect(debugElement.children.length).toBe(2);
// test NgFor // test NgFor
comp.instance.ctxArr = [1, 2]; compFixture.componentInstance.ctxArr = [1, 2];
comp.changeDetectorRef.detectChanges(); compFixture.detectChanges();
expect(debugElement.children.length).toBe(4); expect(debugElement.children.length).toBe(4);
expect(debugElement.children[2].attributes['value']).toBe('1'); expect(debugElement.children[2].attributes['value']).toBe('1');
expect(debugElement.children[3].attributes['value']).toBe('2'); expect(debugElement.children[3].attributes['value']).toBe('2');

View File

@ -6,11 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
// Only needed to satisfy the check in core/src/util/decorators.ts import './init';
// TODO(alexeagle): maybe remove that check?
require('reflect-metadata');
require('zone.js/dist/zone-node.js');
require('zone.js/dist/long-stack-trace-zone.js');
let serializer = require('@angular/compiler/src/i18n/xmb_serializer.js'); let serializer = require('@angular/compiler/src/i18n/xmb_serializer.js');
import * as fs from 'fs'; import * as fs from 'fs';

View File

@ -0,0 +1,18 @@
/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* 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
*/
// Only needed to satisfy the check in core/src/util/decorators.ts
// TODO(alexeagle): maybe remove that check?
require('reflect-metadata');
require('zone.js/dist/zone-node.js');
require('zone.js/dist/long-stack-trace-zone.js');
import {lockRunMode} from '@angular/core';
// Need to lock the mode explicitely as this test is not using Angular's testing framework.
lockRunMode();

View File

@ -6,22 +6,15 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {DebugElement, ReflectiveInjector, getDebugNode, lockRunMode} from '@angular/core'; import './init';
import {BROWSER_APP_PROVIDERS, By} from '@angular/platform-browser'; import {BasicComp} from '../src/basic';
import {serverPlatform} from '@angular/platform-server'; import {CompWithPrecompile} from '../src/precompile';
import {createComponent} from './util';
import {SomeComp} from '../src/precompile';
import {CompWithPrecompileNgFactory} from '../src/precompile.ngfactory';
// Need to lock the mode explicitely as this test is not using Angular's testing framework.
lockRunMode();
describe('content projection', () => { describe('content projection', () => {
it('should support basic content projection', () => { it('should support basic content projection', () => {
const appInjector = var compFixture = createComponent(CompWithPrecompile);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var cf = compFixture.componentInstance.cfr.resolveComponentFactory(BasicComp);
var compWithPrecompile = CompWithPrecompileNgFactory.create(appInjector).instance; expect(cf.componentType).toBe(BasicComp);
var cf = compWithPrecompile.cfr.resolveComponentFactory(SomeComp);
expect(cf.componentType).toBe(SomeComp);
}); });
}); });

View File

@ -6,24 +6,17 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {DebugElement, ReflectiveInjector, getDebugNode, lockRunMode} from '@angular/core'; import './init';
import {BROWSER_APP_PROVIDERS, By} from '@angular/platform-browser'; import {By} from '@angular/platform-browser';
import {serverPlatform} from '@angular/platform-server'; import {CompWithNgContent, ProjectingComp} from '../src/projection';
import {createComponent} from './util';
import {CompWithProjection} from '../src/projection';
import {MainCompNgFactory} from '../src/projection.ngfactory';
// Need to lock the mode explicitely as this test is not using Angular's testing framework.
lockRunMode();
describe('content projection', () => { describe('content projection', () => {
it('should support basic content projection', () => { it('should support basic content projection', () => {
const appInjector = var mainCompFixture = createComponent(ProjectingComp);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector);
var mainComp = MainCompNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(mainComp.location.nativeElement); var debugElement = mainCompFixture.debugElement;
var compWithProjection = debugElement.query(By.directive(CompWithProjection)); var compWithProjection = debugElement.query(By.directive(CompWithNgContent));
expect(compWithProjection.children.length).toBe(1); expect(compWithProjection.children.length).toBe(1);
expect(compWithProjection.children[0].attributes['greeting']).toEqual('Hello world!'); expect(compWithProjection.children[0].attributes['greeting']).toEqual('Hello world!');
}); });

View File

@ -6,37 +6,30 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {DebugElement, QueryList, ReflectiveInjector, getDebugNode, lockRunMode} from '@angular/core'; import './init';
import {BROWSER_APP_PROVIDERS, By} from '@angular/platform-browser'; import {DebugElement, QueryList} from '@angular/core';
import {serverPlatform} from '@angular/platform-server'; import {By} from '@angular/platform-browser';
import {CompForChildQuery, CompWithChildQuery} from '../src/queries'; import {CompForChildQuery, CompWithChildQuery} from '../src/queries';
import {CompWithChildQueryNgFactory} from '../src/queries.ngfactory'; import {createComponent} from './util';
describe('child queries', () => { describe('child queries', () => {
it('should support compiling child queries', () => { it('should support compiling child queries', () => {
const appInjector = var childQueryCompFixture = createComponent(CompWithChildQuery);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = childQueryCompFixture.debugElement;
var childQueryComp = CompWithChildQueryNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(childQueryComp.location.nativeElement);
var compWithChildren = debugElement.query(By.directive(CompWithChildQuery)); var compWithChildren = debugElement.query(By.directive(CompWithChildQuery));
expect(childQueryComp.instance.child).toBeDefined(); expect(childQueryCompFixture.componentInstance.child).toBeDefined();
expect(childQueryComp.instance.child instanceof CompForChildQuery).toBe(true); expect(childQueryCompFixture.componentInstance.child instanceof CompForChildQuery).toBe(true);
}); });
it('should support compiling children queries', () => { it('should support compiling children queries', () => {
const appInjector = var childQueryCompFixture = createComponent(CompWithChildQuery);
ReflectiveInjector.resolveAndCreate(BROWSER_APP_PROVIDERS, serverPlatform().injector); var debugElement = childQueryCompFixture.debugElement;
var childQueryComp = CompWithChildQueryNgFactory.create(appInjector);
var debugElement = <DebugElement>getDebugNode(childQueryComp.location.nativeElement);
var compWithChildren = debugElement.query(By.directive(CompWithChildQuery)); var compWithChildren = debugElement.query(By.directive(CompWithChildQuery));
childQueryComp.changeDetectorRef.detectChanges(); childQueryCompFixture.detectChanges();
expect(childQueryComp.instance.children).toBeDefined(); expect(childQueryCompFixture.componentInstance.children).toBeDefined();
expect(childQueryComp.instance.children instanceof QueryList).toBe(true); expect(childQueryCompFixture.componentInstance.children instanceof QueryList).toBe(true);
}); });
}); });

View File

@ -0,0 +1,29 @@
/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* 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
*/
import {AppModuleFactory, AppModuleRef} from '@angular/core';
import {ComponentFixture} from '@angular/core/testing';
import {serverPlatform} from '@angular/platform-server';
import {MainModuleNgFactory} from '../src/module.ngfactory';
export function createModule<M>(factory: AppModuleFactory<M>): AppModuleRef<M> {
return factory.create(serverPlatform().injector);
}
export function createComponent<C>(
comp: {new (...args: any[]): C},
moduleFactory: AppModuleFactory<any> = null): ComponentFixture<C> {
if (!moduleFactory) {
moduleFactory = MainModuleNgFactory;
}
const moduleRef = createModule(moduleFactory);
const compRef =
moduleRef.componentFactoryResolver.resolveComponentFactory(comp).create(moduleRef.injector);
return new ComponentFixture(compRef, null, null);
}

View File

@ -246,7 +246,7 @@ export class StaticReflector implements ReflectorReader {
let calling = new Map<StaticSymbol, boolean>(); let calling = new Map<StaticSymbol, boolean>();
function simplifyInContext(context: StaticSymbol, value: any, depth: number): any { function simplifyInContext(context: StaticSymbol, value: any, depth: number): any {
function resolveReference(expression: any): StaticSymbol { function resolveReference(context: StaticSymbol, expression: any): StaticSymbol {
let staticSymbol: StaticSymbol; let staticSymbol: StaticSymbol;
if (expression['module']) { if (expression['module']) {
staticSymbol = _this.host.findDeclaration( staticSymbol = _this.host.findDeclaration(
@ -257,25 +257,34 @@ export class StaticReflector implements ReflectorReader {
return staticSymbol; return staticSymbol;
} }
function isOpaqueToken(value: any): boolean { function resolveReferenceValue(staticSymbol: StaticSymbol): any {
let result: any = staticSymbol;
let moduleMetadata = _this.getModuleMetadata(staticSymbol.filePath);
let declarationValue =
moduleMetadata ? moduleMetadata['metadata'][staticSymbol.name] : null;
return declarationValue;
}
function isOpaqueToken(context: StaticSymbol, value: any): boolean {
if (value && value.__symbolic === 'new' && value.expression) { if (value && value.__symbolic === 'new' && value.expression) {
let target = value.expression; let target = value.expression;
if (target.__symbolic == 'reference') { if (target.__symbolic == 'reference') {
return sameSymbol(resolveReference(target), _this.opaqueToken); return sameSymbol(resolveReference(context, target), _this.opaqueToken);
} }
} }
return false; return false;
} }
function simplifyCall(expression: any) { function simplifyCall(expression: any) {
let context: {[name: string]: string}|undefined = undefined; let callContext: {[name: string]: string}|undefined = undefined;
if (expression['__symbolic'] == 'call') { if (expression['__symbolic'] == 'call') {
let target = expression['expression']; let target = expression['expression'];
let targetFunction: any;
if (target && target.__symbolic === 'reference') { if (target && target.__symbolic === 'reference') {
context = {name: target.name}; callContext = {name: target.name};
targetFunction = resolveReferenceValue(resolveReference(context, target));
} }
let targetFunction = simplify(target); if (targetFunction && targetFunction['__symbolic'] == 'function') {
if (targetFunction['__symbolic'] == 'function') {
if (calling.get(targetFunction)) { if (calling.get(targetFunction)) {
throw new Error('Recursion not supported'); throw new Error('Recursion not supported');
} }
@ -309,7 +318,8 @@ export class StaticReflector implements ReflectorReader {
// non-angular decorator, and we should just ignore it. // non-angular decorator, and we should just ignore it.
return {__symbolic: 'ignore'}; return {__symbolic: 'ignore'};
} }
return simplify({__symbolic: 'error', message: 'Function call not supported', context}); return simplify(
{__symbolic: 'error', message: 'Function call not supported', context: callContext});
} }
function simplify(expression: any): any { function simplify(expression: any): any {
@ -421,13 +431,11 @@ export class StaticReflector implements ReflectorReader {
return localValue; return localValue;
} }
} }
staticSymbol = resolveReference(expression); staticSymbol = resolveReference(context, expression);
let result: any = staticSymbol; let result: any = staticSymbol;
let moduleMetadata = _this.getModuleMetadata(staticSymbol.filePath); let declarationValue = resolveReferenceValue(result);
let declarationValue =
moduleMetadata ? moduleMetadata['metadata'][staticSymbol.name] : null;
if (declarationValue) { if (declarationValue) {
if (isOpaqueToken(declarationValue)) { if (isOpaqueToken(staticSymbol, declarationValue)) {
// If the referenced symbol is initalized by a new OpaqueToken we can keep the // If the referenced symbol is initalized by a new OpaqueToken we can keep the
// reference to the symbol. // reference to the symbol.
return staticSymbol; return staticSymbol;
@ -438,7 +446,7 @@ export class StaticReflector implements ReflectorReader {
case 'class': case 'class':
return context; return context;
case 'function': case 'function':
return expression; return context;
case 'new': case 'new':
case 'call': case 'call':
// Determine if the function is a built-in conversion // Determine if the function is a built-in conversion

View File

@ -308,6 +308,13 @@ describe('StaticReflector', () => {
.toEqual(host.getStaticSymbol('/src/extern.d.ts', 'nonExisting')); .toEqual(host.getStaticSymbol('/src/extern.d.ts', 'nonExisting'));
}); });
it('should simplify a function reference as a static symbol', () => {
expect(simplify(
new StaticSymbol('/src/cases', 'myFunction'),
({__symbolic: 'function', parameters: ['a'], value: []})))
.toEqual(host.getStaticSymbol('/src/cases', 'myFunction'));
});
it('should simplify values initialized with a function call', () => { it('should simplify values initialized with a function call', () => {
expect(simplify(new StaticSymbol('/tmp/src/function-reference.ts', ''), { expect(simplify(new StaticSymbol('/tmp/src/function-reference.ts', ''), {
__symbolic: 'reference', __symbolic: 'reference',

View File

@ -48,10 +48,6 @@ export var ValueUnwrapper: typeof t.ValueUnwrapper = r.ValueUnwrapper;
export var TemplateRef_: typeof t.TemplateRef_ = r.TemplateRef_; export var TemplateRef_: typeof t.TemplateRef_ = r.TemplateRef_;
export type RenderDebugInfo = t.RenderDebugInfo; export type RenderDebugInfo = t.RenderDebugInfo;
export var RenderDebugInfo: typeof t.RenderDebugInfo = r.RenderDebugInfo; export var RenderDebugInfo: typeof t.RenderDebugInfo = r.RenderDebugInfo;
export var SecurityContext: typeof t.SecurityContext = r.SecurityContext;
export type SecurityContext = t.SecurityContext;
export var SanitizationService: typeof t.SanitizationService = r.SanitizationService;
export type SanitizationService = t.SanitizationService;
export var createProvider: typeof t.createProvider = r.createProvider; export var createProvider: typeof t.createProvider = r.createProvider;
export var isProviderLiteral: typeof t.isProviderLiteral = r.isProviderLiteral; export var isProviderLiteral: typeof t.isProviderLiteral = r.isProviderLiteral;
export var EMPTY_ARRAY: typeof t.EMPTY_ARRAY = r.EMPTY_ARRAY; export var EMPTY_ARRAY: typeof t.EMPTY_ARRAY = r.EMPTY_ARRAY;
@ -70,15 +66,12 @@ export var castByValue: typeof t.castByValue = r.castByValue;
export type Console = t.Console; export type Console = t.Console;
export var Console: typeof t.Console = r.Console; export var Console: typeof t.Console = r.Console;
export var reflector: t.Reflector = r.reflector; export var reflector: t.Reflector = r.reflector;
export var Reflector: typeof t.Reflector = r.Reflector;
export type Reflector = t.Reflector; export type Reflector = t.Reflector;
export type NoOpAnimationPlayer = t.NoOpAnimationPlayer; export type NoOpAnimationPlayer = t.NoOpAnimationPlayer;
export var NoOpAnimationPlayer: typeof t.NoOpAnimationPlayer = r.NoOpAnimationPlayer; export var NoOpAnimationPlayer: typeof t.NoOpAnimationPlayer = r.NoOpAnimationPlayer;
export type AnimationPlayer = t.AnimationPlayer; export type AnimationPlayer = t.AnimationPlayer;
export var AnimationPlayer: typeof t.AnimationPlayer = r.AnimationPlayer; export var AnimationPlayer: typeof t.AnimationPlayer = r.AnimationPlayer;
export type NoOpAnimationDriver = t.NoOpAnimationDriver;
export var NoOpAnimationDriver: typeof t.NoOpAnimationDriver = r.NoOpAnimationDriver;
export type AnimationDriver = t.AnimationDriver;
export var AnimationDriver: typeof t.AnimationDriver = r.AnimationDriver;
export type AnimationSequencePlayer = t.AnimationSequencePlayer; export type AnimationSequencePlayer = t.AnimationSequencePlayer;
export var AnimationSequencePlayer: typeof t.AnimationSequencePlayer = r.AnimationSequencePlayer; export var AnimationSequencePlayer: typeof t.AnimationSequencePlayer = r.AnimationSequencePlayer;
export type AnimationGroupPlayer = t.AnimationGroupPlayer; export type AnimationGroupPlayer = t.AnimationGroupPlayer;

View File

@ -38,6 +38,7 @@ import {Lexer} from './expression_parser/lexer';
import {ViewResolver} from './view_resolver'; import {ViewResolver} from './view_resolver';
import {DirectiveResolver} from './directive_resolver'; import {DirectiveResolver} from './directive_resolver';
import {PipeResolver} from './pipe_resolver'; import {PipeResolver} from './pipe_resolver';
import {Console, Reflector, reflector, ReflectorReader} from '../core_private';
/** /**
* A set of providers that provide `RuntimeCompiler` and its dependencies to use for * A set of providers that provide `RuntimeCompiler` and its dependencies to use for
@ -45,6 +46,9 @@ import {PipeResolver} from './pipe_resolver';
*/ */
export const COMPILER_PROVIDERS: Array<any|Type|{[k: string]: any}|any[]> = export const COMPILER_PROVIDERS: Array<any|Type|{[k: string]: any}|any[]> =
/*@ts2dart_const*/[ /*@ts2dart_const*/[
{provide: Reflector, useValue: reflector},
{provide: ReflectorReader, useExisting: Reflector},
Console,
Lexer, Lexer,
Parser, Parser,
HtmlParser, HtmlParser,

View File

@ -6,9 +6,9 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {AppModuleFactory, ChangeDetectionStrategy, ChangeDetectorRef, ComponentFactory, ComponentFactoryResolver, ElementRef, Injector, QueryList, RenderComponentType, Renderer, SimpleChange, TemplateRef, ViewContainerRef, ViewEncapsulation} from '@angular/core'; import {AppModuleFactory, ChangeDetectionStrategy, ChangeDetectorRef, ComponentFactory, ComponentFactoryResolver, ElementRef, Injector, QueryList, RenderComponentType, Renderer, SecurityContext, SimpleChange, TemplateRef, ViewContainerRef, ViewEncapsulation} from '@angular/core';
import {AnimationGroupPlayer as AnimationGroupPlayer_, AnimationKeyframe as AnimationKeyframe_, AnimationSequencePlayer as AnimationSequencePlayer_, AnimationStyles as AnimationStyles_, AppElement, AppModuleInjector, AppView, ChangeDetectorStatus, CodegenComponentFactoryResolver, DebugAppView, DebugContext, EMPTY_ARRAY, EMPTY_MAP, NoOpAnimationPlayer as NoOpAnimationPlayer_, SecurityContext, StaticNodeDebugInfo, TemplateRef_, ValueUnwrapper, ViewType, ViewUtils, balanceAnimationKeyframes as impBalanceAnimationKeyframes, castByValue, checkBinding, clearStyles as impClearStyles, collectAndResolveStyles as impCollectAndResolveStyles, devModeEqual, flattenNestedViewRenderNodes, interpolate, prepareFinalAnimationStyles as impBalanceAnimationStyles, pureProxy1, pureProxy10, pureProxy2, pureProxy3, pureProxy4, pureProxy5, pureProxy6, pureProxy7, pureProxy8, pureProxy9, renderStyles as impRenderStyles, uninitialized} from '../core_private'; import {AnimationGroupPlayer as AnimationGroupPlayer_, AnimationKeyframe as AnimationKeyframe_, AnimationSequencePlayer as AnimationSequencePlayer_, AnimationStyles as AnimationStyles_, AppElement, AppModuleInjector, AppView, ChangeDetectorStatus, CodegenComponentFactoryResolver, DebugAppView, DebugContext, EMPTY_ARRAY, EMPTY_MAP, NoOpAnimationPlayer as NoOpAnimationPlayer_, StaticNodeDebugInfo, TemplateRef_, ValueUnwrapper, ViewType, ViewUtils, balanceAnimationKeyframes as impBalanceAnimationKeyframes, castByValue, checkBinding, clearStyles as impClearStyles, collectAndResolveStyles as impCollectAndResolveStyles, devModeEqual, flattenNestedViewRenderNodes, interpolate, prepareFinalAnimationStyles as impBalanceAnimationStyles, pureProxy1, pureProxy10, pureProxy2, pureProxy3, pureProxy4, pureProxy5, pureProxy6, pureProxy7, pureProxy8, pureProxy9, renderStyles as impRenderStyles, uninitialized} from '../core_private';
import {CompileIdentifierMetadata, CompileTokenMetadata} from './compile_metadata'; import {CompileIdentifierMetadata, CompileTokenMetadata} from './compile_metadata';
import {assetUrl} from './util'; import {assetUrl} from './util';

View File

@ -196,29 +196,11 @@ export class CompileMetadataResolver {
throw new BaseException( throw new BaseException(
`Could not compile '${stringify(moduleType)}' because it is not an AppModule.`); `Could not compile '${stringify(moduleType)}' because it is not an AppModule.`);
} }
let providers: any[] = [];
if (meta.providers) {
providers.push(...this.getProvidersMetadata(meta.providers));
}
let directives: cpl.CompileTypeMetadata[] = [];
if (meta.directives) {
directives.push(...flattenArray(meta.directives)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
let pipes: cpl.CompileTypeMetadata[] = [];
if (meta.pipes) {
pipes.push(...flattenArray(meta.pipes)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
let precompile: cpl.CompileTypeMetadata[] = [];
if (meta.precompile) {
precompile.push(...flattenArray(meta.precompile)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
let modules: cpl.CompileTypeMetadata[] = []; let modules: cpl.CompileTypeMetadata[] = [];
let providers: any[] = [];
let directives: cpl.CompileTypeMetadata[] = [];
let pipes: cpl.CompileTypeMetadata[] = [];
let precompile: cpl.CompileTypeMetadata[] = [];
if (meta.modules) { if (meta.modules) {
flattenArray(meta.modules).forEach((moduleType) => { flattenArray(meta.modules).forEach((moduleType) => {
var meta = this.getAppModuleMetadata(moduleType); var meta = this.getAppModuleMetadata(moduleType);
@ -231,6 +213,22 @@ export class CompileMetadataResolver {
}); });
} }
if (meta.providers) {
providers.push(...this.getProvidersMetadata(meta.providers));
}
if (meta.directives) {
directives.push(...flattenArray(meta.directives)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
if (meta.pipes) {
pipes.push(...flattenArray(meta.pipes)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
if (meta.precompile) {
precompile.push(...flattenArray(meta.precompile)
.map(type => this.getTypeMetadata(type, staticTypeModuleUrl(type))));
}
compileMeta = new cpl.CompileAppModuleMetadata({ compileMeta = new cpl.CompileAppModuleMetadata({
type: this.getTypeMetadata(moduleType, staticTypeModuleUrl(moduleType)), type: this.getTypeMetadata(moduleType, staticTypeModuleUrl(moduleType)),
providers: providers, providers: providers,

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {AppModuleFactory, AppModuleMetadata, Compiler, ComponentFactory, ComponentResolver, Injectable} from '@angular/core'; import {AppModuleFactory, AppModuleMetadata, Compiler, ComponentFactory, ComponentResolver, Injectable, Provider} from '@angular/core';
import {BaseException} from '../src/facade/exceptions'; import {BaseException} from '../src/facade/exceptions';
import {ConcreteType, IS_DART, Type, isBlank, isString, stringify} from '../src/facade/lang'; import {ConcreteType, IS_DART, Type, isBlank, isString, stringify} from '../src/facade/lang';
@ -76,6 +76,14 @@ export class RuntimeCompiler implements ComponentResolver, Compiler {
let componentCompilePromises: Promise<any>[] = []; let componentCompilePromises: Promise<any>[] = [];
if (!appModuleFactory || !useCache) { if (!appModuleFactory || !useCache) {
var compileModuleMeta = this._metadataResolver.getAppModuleMetadata(moduleType, metadata); var compileModuleMeta = this._metadataResolver.getAppModuleMetadata(moduleType, metadata);
let boundCompiler = new BoundCompiler(
this, compileModuleMeta.directives.map(dir => dir.type.runtime),
compileModuleMeta.pipes.map((pipe) => pipe.type.runtime));
// Always provide a bound Compiler / ComponentResolver
compileModuleMeta.providers.push(this._metadataResolver.getProviderMetadata(
new Provider(Compiler, {useValue: boundCompiler})));
compileModuleMeta.providers.push(this._metadataResolver.getProviderMetadata(
new Provider(ComponentResolver, {useExisting: Compiler})));
var compileResult = this._appModuleCompiler.compile(compileModuleMeta); var compileResult = this._appModuleCompiler.compile(compileModuleMeta);
compileResult.dependencies.forEach((dep) => { compileResult.dependencies.forEach((dep) => {
let compileResult = this._compileComponent( let compileResult = this._compileComponent(
@ -102,21 +110,18 @@ export class RuntimeCompiler implements ComponentResolver, Compiler {
appModuleFactory, Promise.all(componentCompilePromises).then(() => appModuleFactory)); appModuleFactory, Promise.all(componentCompilePromises).then(() => appModuleFactory));
} }
compileComponentAsync<T>(compType: ConcreteType<T>, {moduleDirectives = [], modulePipes = []}: { compileComponentAsync<T>(compType: ConcreteType<T>): Promise<ComponentFactory<T>> {
moduleDirectives?: ConcreteType<any>[], return this._compileComponent(compType, false, [], []).asyncResult;
modulePipes?: ConcreteType<any>[]
} = {}): Promise<ComponentFactory<T>> {
return this._compileComponent(compType, false, moduleDirectives, modulePipes).asyncResult;
} }
compileComponentSync<T>(compType: ConcreteType<T>, {moduleDirectives = [], modulePipes = []}: { compileComponentSync<T>(compType: ConcreteType<T>): ComponentFactory<T> {
moduleDirectives?: ConcreteType<any>[], return this._compileComponent(compType, true, [], []).syncResult;
modulePipes?: ConcreteType<any>[]
} = {}): ComponentFactory<T> {
return this._compileComponent(compType, true, moduleDirectives, modulePipes).syncResult;
} }
private _compileComponent<T>( /**
* @internal
*/
_compileComponent<T>(
compType: ConcreteType<T>, isSync: boolean, moduleDirectives: ConcreteType<any>[], compType: ConcreteType<T>, isSync: boolean, moduleDirectives: ConcreteType<any>[],
modulePipes: ConcreteType<any>[]): SyncAsyncResult<ComponentFactory<T>> { modulePipes: ConcreteType<any>[]): SyncAsyncResult<ComponentFactory<T>> {
var templates = var templates =
@ -343,3 +348,50 @@ function assertComponent(meta: CompileDirectiveMetadata) {
throw new BaseException(`Could not compile '${meta.type.name}' because it is not a component.`); throw new BaseException(`Could not compile '${meta.type.name}' because it is not a component.`);
} }
} }
/**
* A wrapper around `Compiler` and `ComponentResolver` that
* provides default patform directives / pipes.
*/
class BoundCompiler implements Compiler, ComponentResolver {
constructor(
private _delegate: RuntimeCompiler, private _directives: any[], private _pipes: any[]) {}
resolveComponent(component: Type|string): Promise<ComponentFactory<any>> {
if (isString(component)) {
return PromiseWrapper.reject(
new BaseException(`Cannot resolve component using '${component}'.`), null);
}
return this.compileComponentAsync(<ConcreteType<any>>component);
}
compileComponentAsync<T>(compType: ConcreteType<T>): Promise<ComponentFactory<T>> {
return this._delegate._compileComponent(compType, false, this._directives, this._pipes)
.asyncResult;
}
compileComponentSync<T>(compType: ConcreteType<T>): ComponentFactory<T> {
return this._delegate._compileComponent(compType, true, this._directives, this._pipes)
.syncResult;
}
compileAppModuleSync<T>(moduleType: ConcreteType<T>, metadata: AppModuleMetadata = null):
AppModuleFactory<T> {
return this._delegate.compileAppModuleSync(moduleType, metadata);
}
compileAppModuleAsync<T>(moduleType: ConcreteType<T>, metadata: AppModuleMetadata = null):
Promise<AppModuleFactory<T>> {
return this._delegate.compileAppModuleAsync(moduleType, metadata);
}
/**
* Clears all caches
*/
clearCache(): void { this._delegate.clearCache(); }
/**
* Clears the cache for the given component/appModule.
*/
clearCacheFor(type: Type) { this._delegate.clearCacheFor(type); }
}

View File

@ -6,9 +6,8 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {Injectable} from '@angular/core'; import {Injectable, SecurityContext} from '@angular/core';
import {SecurityContext} from '../../core_private';
import {StringMapWrapper} from '../facade/collection'; import {StringMapWrapper} from '../facade/collection';
import {isPresent} from '../facade/lang'; import {isPresent} from '../facade/lang';

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {SecurityContext} from '../../core_private'; import {SecurityContext} from '@angular/core';
// ================================================================================================= // =================================================================================================
// ================================================================================================= // =================================================================================================

View File

@ -12,7 +12,7 @@ import {AST} from './expression_parser/ast';
import {CompileDirectiveMetadata, CompileTokenMetadata, CompileProviderMetadata,} from './compile_metadata'; import {CompileDirectiveMetadata, CompileTokenMetadata, CompileProviderMetadata,} from './compile_metadata';
import {ParseSourceSpan} from './parse_util'; import {ParseSourceSpan} from './parse_util';
import {SecurityContext} from '../core_private'; import {SecurityContext} from '@angular/core';
/** /**
* An Abstract Syntax Tree node representing part of a parsed Angular template. * An Abstract Syntax Tree node representing part of a parsed Angular template.

View File

@ -6,9 +6,9 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {Inject, Injectable, OpaqueToken, Optional} from '@angular/core'; import {Inject, Injectable, OpaqueToken, Optional, SecurityContext} from '@angular/core';
import {Console, MAX_INTERPOLATION_VALUES, SecurityContext} from '../core_private'; import {Console, MAX_INTERPOLATION_VALUES} from '../core_private';
import {ListWrapper, StringMapWrapper, SetWrapper,} from '../src/facade/collection'; import {ListWrapper, StringMapWrapper, SetWrapper,} from '../src/facade/collection';
import {RegExpWrapper, isPresent, StringWrapper, isBlank, isArray} from '../src/facade/lang'; import {RegExpWrapper, isPresent, StringWrapper, isBlank, isArray} from '../src/facade/lang';

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {EMPTY_STATE as EMPTY_ANIMATION_STATE, LifecycleHooks, SecurityContext, isDefaultChangeDetectionStrategy} from '../../core_private'; import {EMPTY_STATE as EMPTY_ANIMATION_STATE, LifecycleHooks, isDefaultChangeDetectionStrategy} from '../../core_private';
import * as cdAst from '../expression_parser/ast'; import * as cdAst from '../expression_parser/ast';
import {isBlank, isPresent} from '../facade/lang'; import {isBlank, isPresent} from '../facade/lang';
import {Identifiers} from '../identifiers'; import {Identifiers} from '../identifiers';
@ -25,7 +25,7 @@ import {camelCaseToDashCase} from '../util';
import {convertCdExpressionToIr} from './expression_converter'; import {convertCdExpressionToIr} from './expression_converter';
import {CompileBinding} from './compile_binding'; import {CompileBinding} from './compile_binding';
import {BaseException} from '@angular/core'; import {BaseException, SecurityContext} from '@angular/core';
function createBindFieldExpr(exprIndex: number): o.ReadPropExpr { function createBindFieldExpr(exprIndex: number): o.ReadPropExpr {

View File

@ -9,11 +9,10 @@
import {HtmlElementAst} from '@angular/compiler/src/html_ast'; import {HtmlElementAst} from '@angular/compiler/src/html_ast';
import {HtmlParser} from '@angular/compiler/src/html_parser'; import {HtmlParser} from '@angular/compiler/src/html_parser';
import {DomElementSchemaRegistry} from '@angular/compiler/src/schema/dom_element_schema_registry'; import {DomElementSchemaRegistry} from '@angular/compiler/src/schema/dom_element_schema_registry';
import {SecurityContext} from '@angular/core';
import {beforeEach, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '@angular/core/testing/testing_internal'; import {beforeEach, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '@angular/core/testing/testing_internal';
import {browserDetection} from '@angular/platform-browser/testing/browser_util'; import {browserDetection} from '@angular/platform-browser/testing/browser_util';
import {SecurityContext} from '../../core_private';
import {extractSchema} from './schema_extractor'; import {extractSchema} from './schema_extractor';
export function main() { export function main() {

View File

@ -12,10 +12,10 @@ import {ElementSchemaRegistry} from '@angular/compiler/src/schema/element_schema
import {AttrAst, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, DirectiveAst, ElementAst, EmbeddedTemplateAst, NgContentAst, PropertyBindingType, ProviderAstType, ReferenceAst, TemplateAst, TemplateAstVisitor, TextAst, VariableAst, templateVisitAll} from '@angular/compiler/src/template_ast'; import {AttrAst, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, DirectiveAst, ElementAst, EmbeddedTemplateAst, NgContentAst, PropertyBindingType, ProviderAstType, ReferenceAst, TemplateAst, TemplateAstVisitor, TextAst, VariableAst, templateVisitAll} from '@angular/compiler/src/template_ast';
import {TEMPLATE_TRANSFORMS, TemplateParser, splitClasses} from '@angular/compiler/src/template_parser'; import {TEMPLATE_TRANSFORMS, TemplateParser, splitClasses} from '@angular/compiler/src/template_parser';
import {MockSchemaRegistry} from '@angular/compiler/testing'; import {MockSchemaRegistry} from '@angular/compiler/testing';
import {SecurityContext} from '@angular/core';
import {Console} from '@angular/core/src/console'; import {Console} from '@angular/core/src/console';
import {afterEach, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xit} from '@angular/core/testing/testing_internal'; import {afterEach, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xit} from '@angular/core/testing/testing_internal';
import {SecurityContext} from '../core_private';
import {Identifiers, identifierToken} from '../src/identifiers'; import {Identifiers, identifierToken} from '../src/identifiers';
import {DEFAULT_INTERPOLATION_CONFIG, InterpolationConfig} from '../src/interpolation_config'; import {DEFAULT_INTERPOLATION_CONFIG, InterpolationConfig} from '../src/interpolation_config';

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {SecurityContext} from '../core_private'; import {SecurityContext} from '@angular/core';
import {ElementSchemaRegistry} from '../index'; import {ElementSchemaRegistry} from '../index';
import {isPresent} from '../src/facade/lang'; import {isPresent} from '../src/facade/lang';

View File

@ -7,7 +7,7 @@ export './src/core/di.dart' hide ForwardRefFn, resolveForwardRef, forwardRef;
export './src/facade/facade.dart'; export './src/facade/facade.dart';
export './src/core/application_ref.dart' show createPlatform, assertPlatform, export './src/core/application_ref.dart' show createPlatform, assertPlatform,
disposePlatform, getPlatform, disposePlatform, getPlatform,
coreLoadAndBootstrap, coreBootstrap, createNgZone, PlatformRef, ApplicationRef; coreLoadAndBootstrap, coreBootstrap, PlatformRef, ApplicationRef;
export './src/core/application_tokens.dart' show APP_ID, export './src/core/application_tokens.dart' show APP_ID,
APP_INITIALIZER, APP_INITIALIZER,
PACKAGE_ROOT_URL, PACKAGE_ROOT_URL,

View File

@ -14,7 +14,7 @@
export * from './src/metadata'; export * from './src/metadata';
export * from './src/util'; export * from './src/util';
export * from './src/di'; export * from './src/di';
export {createPlatform, assertPlatform, disposePlatform, getPlatform, coreBootstrap, coreLoadAndBootstrap, createNgZone, PlatformRef, ApplicationRef, enableProdMode, lockRunMode, isDevMode} from './src/application_ref'; export {createPlatform, assertPlatform, disposePlatform, getPlatform, coreBootstrap, coreLoadAndBootstrap, PlatformRef, ApplicationRef, enableProdMode, lockRunMode, isDevMode} from './src/application_ref';
export {APP_ID, APP_INITIALIZER, PACKAGE_ROOT_URL, PLATFORM_INITIALIZER} from './src/application_tokens'; export {APP_ID, APP_INITIALIZER, PACKAGE_ROOT_URL, PLATFORM_INITIALIZER} from './src/application_tokens';
export * from './src/zone'; export * from './src/zone';
export * from './src/render'; export * from './src/render';
@ -24,7 +24,7 @@ export * from './src/testability/testability';
export * from './src/change_detection'; export * from './src/change_detection';
export * from './src/platform_directives_and_pipes'; export * from './src/platform_directives_and_pipes';
export * from './src/platform_common_providers'; export * from './src/platform_common_providers';
export * from './src/application_common_providers'; export {APPLICATION_COMMON_PROVIDERS} from './src/application_common_providers';
export {wtfCreateScope, wtfLeave, wtfStartTimeRange, wtfEndTimeRange, WtfScopeFn} from './src/profile/profile'; export {wtfCreateScope, wtfLeave, wtfStartTimeRange, wtfEndTimeRange, WtfScopeFn} from './src/profile/profile';
export {Type} from './src/facade/lang'; export {Type} from './src/facade/lang';
@ -34,3 +34,5 @@ export * from './private_export';
export * from './src/animation/metadata'; export * from './src/animation/metadata';
export {AnimationPlayer} from './src/animation/animation_player'; export {AnimationPlayer} from './src/animation/animation_player';
export {SanitizationService, SecurityContext} from './src/security';

View File

@ -8,7 +8,6 @@
import {Provider} from './index'; import {Provider} from './index';
import {ANY_STATE as ANY_STATE_, DEFAULT_STATE as DEFAULT_STATE_, EMPTY_STATE as EMPTY_STATE_, FILL_STYLE_FLAG as FILL_STYLE_FLAG_} from './src/animation/animation_constants'; import {ANY_STATE as ANY_STATE_, DEFAULT_STATE as DEFAULT_STATE_, EMPTY_STATE as EMPTY_STATE_, FILL_STYLE_FLAG as FILL_STYLE_FLAG_} from './src/animation/animation_constants';
import {AnimationDriver as AnimationDriver_, NoOpAnimationDriver as NoOpAnimationDriver_} from './src/animation/animation_driver';
import {AnimationGroupPlayer as AnimationGroupPlayer_} from './src/animation/animation_group_player'; import {AnimationGroupPlayer as AnimationGroupPlayer_} from './src/animation/animation_group_player';
import {AnimationKeyframe as AnimationKeyframe_} from './src/animation/animation_keyframe'; import {AnimationKeyframe as AnimationKeyframe_} from './src/animation/animation_keyframe';
import {AnimationPlayer as AnimationPlayer_, NoOpAnimationPlayer as NoOpAnimationPlayer_} from './src/animation/animation_player'; import {AnimationPlayer as AnimationPlayer_, NoOpAnimationPlayer as NoOpAnimationPlayer_} from './src/animation/animation_player';
@ -77,10 +76,6 @@ export declare namespace __core_private_types__ {
export var ValueUnwrapper: typeof change_detection_util.ValueUnwrapper; export var ValueUnwrapper: typeof change_detection_util.ValueUnwrapper;
export type RenderDebugInfo = api.RenderDebugInfo; export type RenderDebugInfo = api.RenderDebugInfo;
export var RenderDebugInfo: typeof api.RenderDebugInfo; export var RenderDebugInfo: typeof api.RenderDebugInfo;
export type SecurityContext = security.SecurityContext;
export var SecurityContext: typeof security.SecurityContext;
export type SanitizationService = security.SanitizationService;
export var SanitizationService: typeof security.SanitizationService;
export type TemplateRef_<C> = template_ref.TemplateRef_<C>; export type TemplateRef_<C> = template_ref.TemplateRef_<C>;
export var TemplateRef_: typeof template_ref.TemplateRef_; export var TemplateRef_: typeof template_ref.TemplateRef_;
export var wtfInit: typeof wtf_init.wtfInit; export var wtfInit: typeof wtf_init.wtfInit;
@ -113,10 +108,6 @@ export declare namespace __core_private_types__ {
export var NoOpAnimationPlayer: typeof NoOpAnimationPlayer_; export var NoOpAnimationPlayer: typeof NoOpAnimationPlayer_;
export type AnimationPlayer = AnimationPlayer_; export type AnimationPlayer = AnimationPlayer_;
export var AnimationPlayer: typeof AnimationPlayer_; export var AnimationPlayer: typeof AnimationPlayer_;
export type NoOpAnimationDriver = NoOpAnimationDriver_;
export var NoOpAnimationDriver: typeof NoOpAnimationDriver_;
export type AnimationDriver = AnimationDriver_;
export var AnimationDriver: typeof AnimationDriver_;
export type AnimationSequencePlayer = AnimationSequencePlayer_; export type AnimationSequencePlayer = AnimationSequencePlayer_;
export var AnimationSequencePlayer: typeof AnimationSequencePlayer_; export var AnimationSequencePlayer: typeof AnimationSequencePlayer_;
export type AnimationGroupPlayer = AnimationGroupPlayer_; export type AnimationGroupPlayer = AnimationGroupPlayer_;
@ -164,8 +155,6 @@ export var __core_private__ = {
uninitialized: change_detection_util.uninitialized, uninitialized: change_detection_util.uninitialized,
ValueUnwrapper: change_detection_util.ValueUnwrapper, ValueUnwrapper: change_detection_util.ValueUnwrapper,
RenderDebugInfo: api.RenderDebugInfo, RenderDebugInfo: api.RenderDebugInfo,
SecurityContext: security.SecurityContext,
SanitizationService: security.SanitizationService,
TemplateRef_: template_ref.TemplateRef_, TemplateRef_: template_ref.TemplateRef_,
wtfInit: wtf_init.wtfInit, wtfInit: wtf_init.wtfInit,
ReflectionCapabilities: reflection_capabilities.ReflectionCapabilities, ReflectionCapabilities: reflection_capabilities.ReflectionCapabilities,
@ -191,8 +180,6 @@ export var __core_private__ = {
Reflector: reflection.Reflector, Reflector: reflection.Reflector,
NoOpAnimationPlayer: NoOpAnimationPlayer_, NoOpAnimationPlayer: NoOpAnimationPlayer_,
AnimationPlayer: AnimationPlayer_, AnimationPlayer: AnimationPlayer_,
NoOpAnimationDriver: NoOpAnimationDriver_,
AnimationDriver: AnimationDriver_,
AnimationSequencePlayer: AnimationSequencePlayer_, AnimationSequencePlayer: AnimationSequencePlayer_,
AnimationGroupPlayer: AnimationGroupPlayer_, AnimationGroupPlayer: AnimationGroupPlayer_,
AnimationKeyframe: AnimationKeyframe_, AnimationKeyframe: AnimationKeyframe_,

View File

@ -18,6 +18,18 @@ import {ViewUtils} from './linker/view_utils';
let __unused: Type; // avoid unused import when Type union types are erased let __unused: Type; // avoid unused import when Type union types are erased
export function _componentFactoryResolverFactory() {
return ComponentFactoryResolver.NULL;
}
export function _iterableDiffersFactory() {
return defaultIterableDiffers;
}
export function _keyValueDiffersFactory() {
return defaultKeyValueDiffers;
}
/** /**
* A default set of providers which should be included in any Angular * A default set of providers which should be included in any Angular
* application, regardless of the platform it runs onto. * application, regardless of the platform it runs onto.
@ -27,10 +39,18 @@ export const APPLICATION_COMMON_PROVIDERS: Array<Type|{[k: string]: any}|any[]>
/*@ts2dart_const*/[ /*@ts2dart_const*/[
APPLICATION_CORE_PROVIDERS, APPLICATION_CORE_PROVIDERS,
/* @ts2dart_Provider */ {provide: ComponentResolver, useClass: ReflectorComponentResolver}, /* @ts2dart_Provider */ {provide: ComponentResolver, useClass: ReflectorComponentResolver},
{provide: ComponentFactoryResolver, useValue: ComponentFactoryResolver.NULL}, {provide: ComponentFactoryResolver, useFactory: _componentFactoryResolverFactory, deps: []},
APP_ID_RANDOM_PROVIDER, APP_ID_RANDOM_PROVIDER,
ViewUtils, ViewUtils,
/* @ts2dart_Provider */ {provide: IterableDiffers, useValue: defaultIterableDiffers}, /* @ts2dart_Provider */ {
/* @ts2dart_Provider */ {provide: KeyValueDiffers, useValue: defaultKeyValueDiffers}, provide: IterableDiffers,
useFactory: _iterableDiffersFactory,
deps: []
},
/* @ts2dart_Provider */ {
provide: KeyValueDiffers,
useFactory: _keyValueDiffersFactory,
deps: []
},
/* @ts2dart_Provider */ {provide: DynamicComponentLoader, useClass: DynamicComponentLoader_}, /* @ts2dart_Provider */ {provide: DynamicComponentLoader, useClass: DynamicComponentLoader_},
]; ];

View File

@ -9,23 +9,35 @@
import {ObservableWrapper, PromiseWrapper} from '../src/facade/async'; import {ObservableWrapper, PromiseWrapper} from '../src/facade/async';
import {ListWrapper} from '../src/facade/collection'; import {ListWrapper} from '../src/facade/collection';
import {BaseException, ExceptionHandler, unimplemented} from '../src/facade/exceptions'; import {BaseException, ExceptionHandler, unimplemented} from '../src/facade/exceptions';
import {IS_DART, Type, isBlank, isPresent, isPromise} from '../src/facade/lang'; import {ConcreteType, IS_DART, Type, isBlank, isPresent, isPromise} from '../src/facade/lang';
import {APP_INITIALIZER, PLATFORM_INITIALIZER} from './application_tokens'; import {APP_INITIALIZER, PLATFORM_INITIALIZER} from './application_tokens';
import {ChangeDetectorRef} from './change_detection/change_detector_ref'; import {ChangeDetectorRef} from './change_detection/change_detector_ref';
import {Console} from './console'; import {Console} from './console';
import {Injectable, Injector} from './di'; import {Inject, Injectable, Injector, Optional, OptionalMetadata, SkipSelf, SkipSelfMetadata, forwardRef} from './di';
import {ComponentFactory, ComponentRef} from './linker/component_factory'; import {ComponentFactory, ComponentRef} from './linker/component_factory';
import {ComponentFactoryResolver} from './linker/component_factory_resolver';
import {ComponentResolver} from './linker/component_resolver'; import {ComponentResolver} from './linker/component_resolver';
import {WtfScopeFn, wtfCreateScope, wtfLeave} from './profile/profile'; import {WtfScopeFn, wtfCreateScope, wtfLeave} from './profile/profile';
import {Testability, TestabilityRegistry} from './testability/testability'; import {Testability, TestabilityRegistry} from './testability/testability';
import {NgZone, NgZoneError} from './zone/ng_zone'; import {NgZone, NgZoneError} from './zone/ng_zone';
/** /**
* Create an Angular zone. * Create an Angular zone.
* @experimental * @experimental
*/ */
export function createNgZone(): NgZone { export function createNgZone(parent: NgZone): NgZone {
// If an NgZone is already present in the parent injector,
// use that one. Creating the NgZone in the same injector as the
// application is dangerous as some services might get created before
// the NgZone has been created.
// We keep the NgZone factory in the application providers for
// backwards compatibility for now though.
if (parent) {
return parent;
}
return new NgZone({enableLongStackTrace: isDevMode()}); return new NgZone({enableLongStackTrace: isDevMode()});
} }
@ -279,7 +291,7 @@ export abstract class ApplicationRef {
* ### Example * ### Example
* {@example core/ts/platform/platform.ts region='longform'} * {@example core/ts/platform/platform.ts region='longform'}
*/ */
abstract bootstrap<C>(componentFactory: ComponentFactory<C>): ComponentRef<C>; abstract bootstrap<C>(componentFactory: ComponentFactory<C>|ConcreteType<C>): ComponentRef<C>;
/** /**
* Retrieve the application {@link Injector}. * Retrieve the application {@link Injector}.
@ -334,18 +346,19 @@ export class ApplicationRef_ extends ApplicationRef {
/** @internal */ /** @internal */
private _enforceNoNewChanges: boolean = false; private _enforceNoNewChanges: boolean = false;
private _exceptionHandler: ExceptionHandler;
private _asyncInitDonePromise: Promise<any>; private _asyncInitDonePromise: Promise<any>;
private _asyncInitDone: boolean; private _asyncInitDone: boolean;
constructor(private _platform: PlatformRef_, private _zone: NgZone, private _injector: Injector) { constructor(
private _platform: PlatformRef_, private _zone: NgZone, private _console: Console,
private _injector: Injector, private _exceptionHandler: ExceptionHandler,
private _componentFactoryResolver: ComponentFactoryResolver,
@Optional() private _testabilityRegistry: TestabilityRegistry,
@Optional() private _testability: Testability,
@Optional() @Inject(APP_INITIALIZER) inits: Function[]) {
super(); super();
var zone: NgZone = _injector.get(NgZone);
this._enforceNoNewChanges = isDevMode(); this._enforceNoNewChanges = isDevMode();
zone.run(() => { this._exceptionHandler = _injector.get(ExceptionHandler); });
this._asyncInitDonePromise = this.run(() => { this._asyncInitDonePromise = this.run(() => {
let inits: Function[] = _injector.get(APP_INITIALIZER, null);
var asyncInitResults: Promise<any>[] = []; var asyncInitResults: Promise<any>[] = [];
var asyncInitDonePromise: Promise<any>; var asyncInitDonePromise: Promise<any>;
if (isPresent(inits)) { if (isPresent(inits)) {
@ -366,7 +379,7 @@ export class ApplicationRef_ extends ApplicationRef {
} }
return asyncInitDonePromise; return asyncInitDonePromise;
}); });
ObservableWrapper.subscribe(zone.onError, (error: NgZoneError) => { ObservableWrapper.subscribe(this._zone.onError, (error: NgZoneError) => {
this._exceptionHandler.call(error.error, error.stackTrace); this._exceptionHandler.call(error.error, error.stackTrace);
}); });
ObservableWrapper.subscribe( ObservableWrapper.subscribe(
@ -390,7 +403,6 @@ export class ApplicationRef_ extends ApplicationRef {
waitForAsyncInitializers(): Promise<any> { return this._asyncInitDonePromise; } waitForAsyncInitializers(): Promise<any> { return this._asyncInitDonePromise; }
run(callback: Function): any { run(callback: Function): any {
var zone = this.injector.get(NgZone);
var result: any; var result: any;
// Note: Don't use zone.runGuarded as we want to know about // Note: Don't use zone.runGuarded as we want to know about
// the thrown exception! // the thrown exception!
@ -398,7 +410,7 @@ export class ApplicationRef_ extends ApplicationRef {
// of `zone.run` as Dart swallows rejected promises // of `zone.run` as Dart swallows rejected promises
// via the onError callback of the promise. // via the onError callback of the promise.
var completer = PromiseWrapper.completer(); var completer = PromiseWrapper.completer();
zone.run(() => { this._zone.run(() => {
try { try {
result = callback(); result = callback();
if (isPromise(result)) { if (isPromise(result)) {
@ -417,12 +429,19 @@ export class ApplicationRef_ extends ApplicationRef {
return isPromise(result) ? completer.promise : result; return isPromise(result) ? completer.promise : result;
} }
bootstrap<C>(componentFactory: ComponentFactory<C>): ComponentRef<C> { bootstrap<C>(componentOrFactory: ComponentFactory<C>|ConcreteType<C>): ComponentRef<C> {
if (!this._asyncInitDone) { if (!this._asyncInitDone) {
throw new BaseException( throw new BaseException(
'Cannot bootstrap as there are still asynchronous initializers running. Wait for them using waitForAsyncInitializers().'); 'Cannot bootstrap as there are still asynchronous initializers running. Wait for them using waitForAsyncInitializers().');
} }
return this.run(() => { return this.run(() => {
let componentFactory: ComponentFactory<C>;
if (componentOrFactory instanceof ComponentFactory) {
componentFactory = componentOrFactory;
} else {
componentFactory =
this._componentFactoryResolver.resolveComponentFactory(componentOrFactory);
}
this._rootComponentTypes.push(componentFactory.componentType); this._rootComponentTypes.push(componentFactory.componentType);
var compRef = componentFactory.create(this._injector, [], componentFactory.selector); var compRef = componentFactory.create(this._injector, [], componentFactory.selector);
compRef.onDestroy(() => { this._unloadComponent(compRef); }); compRef.onDestroy(() => { this._unloadComponent(compRef); });
@ -433,11 +452,10 @@ export class ApplicationRef_ extends ApplicationRef {
} }
this._loadComponent(compRef); this._loadComponent(compRef);
let c: Console = this._injector.get(Console);
if (isDevMode()) { if (isDevMode()) {
let prodDescription = IS_DART ? 'Production mode is disabled in Dart.' : let prodDescription = IS_DART ? 'Production mode is disabled in Dart.' :
'Call enableProdMode() to enable the production mode.'; 'Call enableProdMode() to enable the production mode.';
c.log(`Angular 2 is running in the development mode. ${prodDescription}`); this._console.log(`Angular 2 is running in the development mode. ${prodDescription}`);
} }
return compRef; return compRef;
}); });
@ -500,7 +518,11 @@ export const PLATFORM_CORE_PROVIDERS =
]; ];
export const APPLICATION_CORE_PROVIDERS = /*@ts2dart_const*/[ export const APPLICATION_CORE_PROVIDERS = /*@ts2dart_const*/[
/* @ts2dart_Provider */ {provide: NgZone, useFactory: createNgZone, deps: [] as any}, /* @ts2dart_Provider */ {
provide: NgZone,
useFactory: createNgZone,
deps: <any>[[new SkipSelfMetadata(), new OptionalMetadata(), NgZone]]
},
ApplicationRef_, ApplicationRef_,
/* @ts2dart_Provider */ {provide: ApplicationRef, useExisting: ApplicationRef_}, /* @ts2dart_Provider */ {provide: ApplicationRef, useExisting: ApplicationRef_},
]; ];

View File

@ -23,7 +23,7 @@ import {OpaqueToken} from './di';
*/ */
export const APP_ID: any = /*@ts2dart_const*/ new OpaqueToken('AppId'); export const APP_ID: any = /*@ts2dart_const*/ new OpaqueToken('AppId');
function _appIdRandomProviderFactory() { export function _appIdRandomProviderFactory() {
return `${_randomChar()}${_randomChar()}${_randomChar()}`; return `${_randomChar()}${_randomChar()}${_randomChar()}`;
} }
@ -35,7 +35,7 @@ export const APP_ID_RANDOM_PROVIDER =
/*@ts2dart_const*/ /* @ts2dart_Provider */ { /*@ts2dart_const*/ /* @ts2dart_Provider */ {
provide: APP_ID, provide: APP_ID,
useFactory: _appIdRandomProviderFactory, useFactory: _appIdRandomProviderFactory,
deps: [] as any deps: <any[]>[]
}; };
function _randomChar(): string { function _randomChar(): string {

View File

@ -18,16 +18,17 @@ import {ComponentFactory} from './component_factory';
* Low-level service for running the angular compiler duirng runtime * Low-level service for running the angular compiler duirng runtime
* to create {@link ComponentFactory}s, which * to create {@link ComponentFactory}s, which
* can later be used to create and render a Component instance. * can later be used to create and render a Component instance.
*
* Each `@AppModule` provides an own `Compiler` to its injector,
* that will use the directives/pipes of the app module for compilation
* of components.
* @stable * @stable
*/ */
export class Compiler { export class Compiler {
/** /**
* Loads the template and styles of a component and returns the associated `ComponentFactory`. * Loads the template and styles of a component and returns the associated `ComponentFactory`.
*/ */
compileComponentAsync<T>(component: ConcreteType<T>, {moduleDirectives = [], modulePipes = []}: { compileComponentAsync<T>(component: ConcreteType<T>): Promise<ComponentFactory<T>> {
moduleDirectives?: ConcreteType<any>[],
modulePipes?: ConcreteType<any>[]
} = {}): Promise<ComponentFactory<T>> {
throw new BaseException( throw new BaseException(
`Runtime compiler is not loaded. Tried to compile ${stringify(component)}`); `Runtime compiler is not loaded. Tried to compile ${stringify(component)}`);
} }
@ -35,10 +36,7 @@ export class Compiler {
* Compiles the given component. All templates have to be either inline or compiled via * Compiles the given component. All templates have to be either inline or compiled via
* `compileComponentAsync` before. * `compileComponentAsync` before.
*/ */
compileComponentSync<T>(component: ConcreteType<T>, {moduleDirectives = [], modulePipes = []}: { compileComponentSync<T>(component: ConcreteType<T>): ComponentFactory<T> {
moduleDirectives?: ConcreteType<any>[],
modulePipes?: ConcreteType<any>[]
} = {}): ComponentFactory<T> {
throw new BaseException( throw new BaseException(
`Runtime compiler is not loaded. Tried to compile ${stringify(component)}`); `Runtime compiler is not loaded. Tried to compile ${stringify(component)}`);
} }

View File

@ -28,7 +28,6 @@ import {AnimationPlayer} from '../animation/animation_player';
import {AnimationGroupPlayer} from '../animation/animation_group_player'; import {AnimationGroupPlayer} from '../animation/animation_group_player';
import {AnimationKeyframe} from '../animation/animation_keyframe'; import {AnimationKeyframe} from '../animation/animation_keyframe';
import {AnimationStyles} from '../animation/animation_styles'; import {AnimationStyles} from '../animation/animation_styles';
import {AnimationDriver} from '../animation/animation_driver';
import {ActiveAnimationPlayersMap} from '../animation/active_animation_players_map'; import {ActiveAnimationPlayersMap} from '../animation/active_animation_players_map';
var _scope_check: WtfScopeFn = wtfCreateScope(`AppView#check(ascii id)`); var _scope_check: WtfScopeFn = wtfCreateScope(`AppView#check(ascii id)`);

View File

@ -25,8 +25,7 @@ export enum SecurityContext {
} }
/** /**
* SanitizationService is used by the views to sanitize potentially dangerous values. This is a * SanitizationService is used by the views to sanitize potentially dangerous values.
* private API, use code should only refer to DomSanitizationService.
* *
* @stable * @stable
*/ */

View File

@ -9,16 +9,16 @@
import {NgIf} from '@angular/common'; import {NgIf} from '@angular/common';
import {CompilerConfig} from '@angular/compiler'; import {CompilerConfig} from '@angular/compiler';
import {TestComponentBuilder} from '@angular/compiler/testing'; import {TestComponentBuilder} from '@angular/compiler/testing';
import {AnimationDriver} from '@angular/platform-browser/src/dom/animation_driver';
import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter'; import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter';
import {MockAnimationDriver} from '@angular/platform-browser/testing/mock_animation_driver';
import {Component} from '../../index'; import {Component} from '../../index';
import {DEFAULT_STATE} from '../../src/animation/animation_constants'; import {DEFAULT_STATE} from '../../src/animation/animation_constants';
import {AnimationDriver} from '../../src/animation/animation_driver';
import {AnimationEntryMetadata, animate, group, keyframes, sequence, state, style, transition, trigger} from '../../src/animation/metadata'; import {AnimationEntryMetadata, animate, group, keyframes, sequence, state, style, transition, trigger} from '../../src/animation/metadata';
import {AUTO_STYLE} from '../../src/animation/metadata'; import {AUTO_STYLE} from '../../src/animation/metadata';
import {IS_DART, isArray, isPresent} from '../../src/facade/lang'; import {IS_DART, isArray, isPresent} from '../../src/facade/lang';
import {fakeAsync, flushMicrotasks, tick} from '../../testing'; import {fakeAsync, flushMicrotasks, tick} from '../../testing';
import {MockAnimationDriver} from '../../testing/animation/mock_animation_driver';
import {AsyncTestCompleter, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '../../testing/testing_internal'; import {AsyncTestCompleter, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '../../testing/testing_internal';
export function main() { export function main() {

View File

@ -11,7 +11,7 @@ import {AsyncTestCompleter} from '@angular/core/testing/testing_internal';
import {Type} from '@angular/core'; import {Type} from '@angular/core';
import {SpyChangeDetectorRef} from './spies'; import {SpyChangeDetectorRef} from './spies';
import {ApplicationRef_, ApplicationRef, PLATFORM_CORE_PROVIDERS, APPLICATION_CORE_PROVIDERS} from '@angular/core/src/application_ref'; import {ApplicationRef_, ApplicationRef, PLATFORM_CORE_PROVIDERS, APPLICATION_CORE_PROVIDERS} from '@angular/core/src/application_ref';
import {Injector, APP_INITIALIZER, Component, ReflectiveInjector, coreLoadAndBootstrap, PlatformRef, createPlatform, disposePlatform, ComponentResolver, ChangeDetectorRef} from '@angular/core'; import {Injector, APP_INITIALIZER, Component, ReflectiveInjector, coreLoadAndBootstrap, PlatformRef, createPlatform, disposePlatform, ComponentResolver, ComponentFactoryResolver, ChangeDetectorRef} from '@angular/core';
import {Console} from '@angular/core/src/console'; import {Console} from '@angular/core/src/console';
import {BaseException} from '../src/facade/exceptions'; import {BaseException} from '../src/facade/exceptions';
import {PromiseWrapper, PromiseCompleter, TimerWrapper} from '../src/facade/async'; import {PromiseWrapper, PromiseCompleter, TimerWrapper} from '../src/facade/async';
@ -40,7 +40,7 @@ export function main() {
APPLICATION_CORE_PROVIDERS, {provide: Console, useValue: new _MockConsole()}, APPLICATION_CORE_PROVIDERS, {provide: Console, useValue: new _MockConsole()},
{provide: ExceptionHandler, useValue: new ExceptionHandler(errorLogger, false)}, {provide: ExceptionHandler, useValue: new ExceptionHandler(errorLogger, false)},
{provide: ComponentResolver, useValue: new _MockComponentResolver(someCompFactory)}, {provide: ComponentResolver, useValue: new _MockComponentResolver(someCompFactory)},
providers {provide: ComponentFactoryResolver, useValue: ComponentFactoryResolver.NULL}, providers
], ],
platform.injector); platform.injector);
return appInjector.get(ApplicationRef); return appInjector.get(ApplicationRef);

View File

@ -1,8 +1,8 @@
import {LowerCasePipe, NgIf} from '@angular/common'; import {LowerCasePipe, NgIf} from '@angular/common';
import {CompilerConfig} from '@angular/compiler'; import {CompilerConfig} from '@angular/compiler';
import {AppModule, AppModuleMetadata, Compiler, Component, ComponentFactoryResolver, ComponentRef, DebugElement, Host, Inject, Injectable, Injector, OpaqueToken, Optional, Provider, SelfMetadata, SkipSelf, SkipSelfMetadata, forwardRef, getDebugNode, provide} from '@angular/core'; import {AppModule, AppModuleMetadata, Compiler, Component, ComponentFactoryResolver, ComponentRef, ComponentResolver, DebugElement, Host, Inject, Injectable, Injector, OpaqueToken, Optional, Provider, SelfMetadata, SkipSelf, SkipSelfMetadata, forwardRef, getDebugNode, provide} from '@angular/core';
import {ComponentFixture} from '@angular/core/testing'; import {ComponentFixture} from '@angular/core/testing';
import {beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '@angular/core/testing/testing_internal'; import {AsyncTestCompleter, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '@angular/core/testing/testing_internal';
import {BaseException} from '../../src/facade/exceptions'; import {BaseException} from '../../src/facade/exceptions';
import {ConcreteType, IS_DART, Type, stringify} from '../../src/facade/lang'; import {ConcreteType, IS_DART, Type, stringify} from '../../src/facade/lang';
@ -195,6 +195,26 @@ function declareTests({useJit}: {useJit: boolean}) {
})); }));
checkNgIfAndLowerCasePipe(compFixture, compFixture.debugElement.children[0]); checkNgIfAndLowerCasePipe(compFixture, compFixture.debugElement.children[0]);
}); });
it('should provide a Compiler instance that uses the directives/pipes of the module', () => {
let appModule = compiler.compileAppModuleSync(ModuleWithDirectivesAndPipes).create();
let boundCompiler: Compiler = appModule.injector.get(Compiler);
var cf = boundCompiler.compileComponentSync(CompUsingModuleDirectiveAndPipe);
let compFixture = new ComponentFixture(cf.create(injector), null, false);
checkNgIfAndLowerCasePipe(compFixture, compFixture.debugElement);
});
it('should provide a ComponentResolver instance that uses the directives/pipes of the module',
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
let appModule = compiler.compileAppModuleSync(ModuleWithDirectivesAndPipes).create();
let boundCompiler: ComponentResolver = appModule.injector.get(ComponentResolver);
boundCompiler.resolveComponent(CompUsingModuleDirectiveAndPipe).then((cf) => {
let compFixture = new ComponentFixture(cf.create(injector), null, false);
checkNgIfAndLowerCasePipe(compFixture, compFixture.debugElement);
async.done();
});
}));
}); });
describe('providers', function() { describe('providers', function() {
@ -471,6 +491,19 @@ function declareTests({useJit}: {useJit: boolean}) {
expect(injector.get('a')).toBe('aValue'); expect(injector.get('a')).toBe('aValue');
expect(injector.get('someToken')).toBe('someValue'); expect(injector.get('someToken')).toBe('someValue');
}); });
it('should override the providers of nested modules', () => {
var injector = compiler
.compileAppModuleSync(
SomeModule, new AppModuleMetadata({
providers: [{provide: 'someToken', useValue: 'someNewValue'}],
modules: [ModuleWithProvider]
}))
.create()
.injector;
expect(injector.get('someToken')).toBe('someNewValue');
});
}); });
}); });

View File

@ -40,6 +40,10 @@ export function main() {
var renderLog: RenderLog; var renderLog: RenderLog;
var directiveLog: DirectiveLog; var directiveLog: DirectiveLog;
function createCompFixture<T>(template: string): ComponentFixture<TestComponent>;
function createCompFixture<T>(template: string, compType: ConcreteType<T>): ComponentFixture<T>;
function createCompFixture<T>(
template: string, compType: ConcreteType<T>, _tcb: TestComponentBuilder): ComponentFixture<T>;
function createCompFixture<T>( function createCompFixture<T>(
template: string, compType: ConcreteType<T> = <any>TestComponent, template: string, compType: ConcreteType<T> = <any>TestComponent,
_tcb: TestComponentBuilder = null): ComponentFixture<T> { _tcb: TestComponentBuilder = null): ComponentFixture<T> {
@ -58,19 +62,23 @@ export function main() {
return nodes.map(node => node.injector.get(dirType)); return nodes.map(node => node.injector.get(dirType));
} }
function _bindSimpleProp<T>(bindAttr: string): ComponentFixture<TestComponent>;
function _bindSimpleProp<T>(bindAttr: string, compType: ConcreteType<T>): ComponentFixture<T>;
function _bindSimpleProp<T>( function _bindSimpleProp<T>(
bindAttr: string, compType: ConcreteType<T> = <any>TestComponent): ComponentFixture<T> { bindAttr: string, compType: ConcreteType<T> = <any>TestComponent): ComponentFixture<T> {
var template = `<div ${bindAttr}></div>`; var template = `<div ${bindAttr}></div>`;
return createCompFixture(template, compType); return createCompFixture(template, compType);
} }
function _bindSimpleValue(expression: any): ComponentFixture<TestComponent>;
function _bindSimpleValue<T>(expression: any, compType: ConcreteType<T>): ComponentFixture<T>;
function _bindSimpleValue<T>( function _bindSimpleValue<T>(
expression: any, compType: ConcreteType<T> = <any>TestComponent): ComponentFixture<T> { expression: any, compType: ConcreteType<T> = <any>TestComponent): ComponentFixture<T> {
return _bindSimpleProp(`[someProp]='${expression}'`, compType); return _bindSimpleProp(`[someProp]='${expression}'`, compType);
} }
function _bindAndCheckSimpleValue<T>( function _bindAndCheckSimpleValue(
expression: any, compType: ConcreteType<T> = <any>TestComponent): string[] { expression: any, compType: ConcreteType<any> = TestComponent): string[] {
var ctx = _bindSimpleValue(expression, compType); var ctx = _bindSimpleValue(expression, compType);
ctx.detectChanges(false); ctx.detectChanges(false);
return renderLog.log; return renderLog.log;

View File

@ -28,7 +28,7 @@ export class ComponentFixture<T> {
/** /**
* The instance of the root component class. * The instance of the root component class.
*/ */
componentInstance: any; componentInstance: T;
/** /**
* The native element at the root of the component. * The native element at the root of the component.

View File

@ -13,7 +13,6 @@ import {Math, global, isFunction, isPromise} from '../src/facade/lang';
import {AsyncTestCompleter} from './async_test_completer'; import {AsyncTestCompleter} from './async_test_completer';
import {getTestInjector, inject} from './test_injector'; import {getTestInjector, inject} from './test_injector';
export {MockAnimationDriver} from './animation/mock_animation_driver';
export {MockAnimationPlayer} from './animation/mock_animation_player'; export {MockAnimationPlayer} from './animation/mock_animation_player';
export {AsyncTestCompleter} from './async_test_completer'; export {AsyncTestCompleter} from './async_test_completer';
export {inject} from './test_injector'; export {inject} from './test_injector';

View File

@ -9,7 +9,7 @@
import {NG_VALIDATORS} from '@angular/common'; import {NG_VALIDATORS} from '@angular/common';
import {bootstrap} from '@angular/platform-browser-dynamic'; import {bootstrap} from '@angular/platform-browser-dynamic';
let MyApp: Function = null; class MyApp {}
let myValidator: any = null; let myValidator: any = null;
// #docregion ng_validators // #docregion ng_validators

View File

@ -8,16 +8,18 @@
import {COMMON_DIRECTIVES, COMMON_PIPES} from '@angular/common'; import {COMMON_DIRECTIVES, COMMON_PIPES} from '@angular/common';
import {COMPILER_PROVIDERS, CompilerConfig, XHR} from '@angular/compiler'; import {COMPILER_PROVIDERS, CompilerConfig, XHR} from '@angular/compiler';
import {ApplicationRef, ComponentRef, PLATFORM_DIRECTIVES, PLATFORM_PIPES, ReflectiveInjector, Type, coreLoadAndBootstrap} from '@angular/core'; import {AppModule, AppModuleRef, ApplicationRef, Compiler, ComponentRef, ComponentResolver, ExceptionHandler, PLATFORM_DIRECTIVES, PLATFORM_PIPES, ReflectiveInjector, Type, coreLoadAndBootstrap, isDevMode, lockRunMode} from '@angular/core';
import {BROWSER_APP_PROVIDERS, WORKER_APP_APPLICATION_PROVIDERS, WORKER_SCRIPT, WORKER_UI_APPLICATION_PROVIDERS, browserPlatform, workerAppPlatform, workerUiPlatform} from '@angular/platform-browser'; import {BROWSER_APP_PROVIDERS, BrowserModule, WORKER_APP_APPLICATION_PROVIDERS, WORKER_SCRIPT, WORKER_UI_APPLICATION_PROVIDERS, bootstrapModuleFactory, browserPlatform, workerAppPlatform, workerUiPlatform} from '@angular/platform-browser';
import {ReflectionCapabilities, reflector} from './core_private'; import {ReflectionCapabilities, reflector} from './core_private';
import {getDOM, initDomAdapter} from './platform_browser_private';
import {PromiseWrapper} from './src/facade/async'; import {PromiseWrapper} from './src/facade/async';
import {isPresent} from './src/facade/lang'; import {ConcreteType, isPresent, stringify} from './src/facade/lang';
import {CachedXHR} from './src/xhr/xhr_cache'; import {CachedXHR} from './src/xhr/xhr_cache';
import {XHRImpl} from './src/xhr/xhr_impl'; import {XHRImpl} from './src/xhr/xhr_impl';
/** /**
* @experimental * @experimental
*/ */
@ -41,7 +43,55 @@ export const BROWSER_APP_COMPILER_PROVIDERS: Array<any /*Type | Provider | any[]
export const CACHED_TEMPLATE_PROVIDER: Array<any /*Type | Provider | any[]*/> = export const CACHED_TEMPLATE_PROVIDER: Array<any /*Type | Provider | any[]*/> =
[{provide: XHR, useClass: CachedXHR}]; [{provide: XHR, useClass: CachedXHR}];
function _initGlobals() {
lockRunMode();
initDomAdapter();
reflector.reflectionCapabilities = new ReflectionCapabilities();
}
/**
* Creates the runtime compiler for the browser.
*
* @stable
*/
export function browserCompiler({useDebug, useJit = true, providers = []}: {
useDebug?: boolean,
useJit?: boolean,
providers?: Array<any /*Type | Provider | any[]*/>
} = {}): Compiler {
_initGlobals();
if (useDebug === undefined) {
useDebug = isDevMode();
}
const injector = ReflectiveInjector.resolveAndCreate([
COMPILER_PROVIDERS, {
provide: CompilerConfig,
useValue: new CompilerConfig({genDebugInfo: useDebug, useJit: useJit})
},
{provide: XHR, useClass: XHRImpl}, providers ? providers : []
]);
return injector.get(Compiler);
}
/**
* Creates an instance of an `@AppModule` for the browser platform.
*
* ## Simple Example
*
* ```typescript
* @AppModule({
* modules: [BrowserModule]
* })
* class MyModule {}
*
* let moduleRef = bootstrapModule(MyModule);
* ```
* @stable
*/
export function bootstrapModule<M>(
moduleType: ConcreteType<M>, compiler: Compiler = browserCompiler()): Promise<AppModuleRef<M>> {
return compiler.compileAppModuleAsync(moduleType).then(bootstrapModuleFactory);
}
/** /**
* Bootstrapping for Angular applications. * Bootstrapping for Angular applications.
@ -102,30 +152,108 @@ export const CACHED_TEMPLATE_PROVIDER: Array<any /*Type | Provider | any[]*/> =
* applications on a page, Angular treats each application injector's services as private * applications on a page, Angular treats each application injector's services as private
* to that application. * to that application.
* *
* ## API * ## API (version 1)
* *
* - `appComponentType`: The root component which should act as the application. This is * - `appComponentType`: The root component which should act as the application. This is
* a reference to a `Type` which is annotated with `@Component(...)`. * a reference to a `Type` which is annotated with `@Component(...)`.
* - `customProviders`: An additional set of providers that can be added to the * - `customProviders`: An additional set of providers that can be added to the
* app injector to override default injection behavior. * app injector to override default injection behavior.
* *
* ## API (version 2)
* - `appComponentType`: The root component which should act as the application. This is
* a reference to a `Type` which is annotated with `@Component(...)`.
* - `providers`, `directives`, `pipes`, `modules`, `precompile`: Defines the properties
* of the dynamically created module that is used to bootstrap the module.
*
* Returns a `Promise` of {@link ComponentRef}. * Returns a `Promise` of {@link ComponentRef}.
* *
* @experimental This api cannot be used with the offline compiler and thus is still subject to * @experimental This api cannot be used with the offline compiler and thus is still subject to
* change. * change.
*/ */
export function bootstrap( // Note: We are using typescript overloads here to have 2 function signatures!
appComponentType: Type, export function bootstrap<C>(
customProviders?: Array<any /*Type | Provider | any[]*/>): Promise<ComponentRef<any>> { appComponentType: ConcreteType<C>,
reflector.reflectionCapabilities = new ReflectionCapabilities(); customProviders?: Array<any /*Type | Provider | any[]*/>): Promise<ComponentRef<C>>;
let providers = [ export function bootstrap<C>(
BROWSER_APP_PROVIDERS, BROWSER_APP_COMPILER_PROVIDERS, appComponentType: ConcreteType<C>,
isPresent(customProviders) ? customProviders : [] {providers, directives, pipes, modules, precompile, compiler}?: {
]; providers?: Array<any /*Type | Provider | any[]*/>,
var appInjector = ReflectiveInjector.resolveAndCreate(providers, browserPlatform().injector); directives?: any[],
return coreLoadAndBootstrap(appComponentType, appInjector); pipes?: any[],
modules?: any[],
precompile?: any[],
compiler?: Compiler
}): Promise<ComponentRef<C>>;
export function bootstrap<C>(
appComponentType: ConcreteType<C>,
customProvidersOrDynamicModule?: Array<any /*Type | Provider | any[]*/>| {
providers: Array<any /*Type | Provider | any[]*/>,
directives: any[],
pipes: any[],
modules: any[],
precompile: any[],
compiler: Compiler
}): Promise<ComponentRef<C>> {
_initGlobals();
let compiler: Compiler;
let compilerProviders: any = [];
let providers: any[] = [];
let directives: any[] = [];
let pipes: any[] = [];
let modules: any[] = [];
let precompile: any[] = [];
if (customProvidersOrDynamicModule instanceof Array) {
providers = customProvidersOrDynamicModule;
} else if (customProvidersOrDynamicModule) {
providers = normalizeArray(customProvidersOrDynamicModule.providers);
directives = normalizeArray(customProvidersOrDynamicModule.directives);
pipes = normalizeArray(customProvidersOrDynamicModule.pipes);
modules = normalizeArray(customProvidersOrDynamicModule.modules);
precompile = normalizeArray(customProvidersOrDynamicModule.precompile);
compiler = customProvidersOrDynamicModule.compiler;
}
if (providers && providers.length > 0) {
// Note: This is a hack to still support the old way
// of configuring platform directives / pipes and the compiler xhr.
// This will soon be deprecated!
let inj = ReflectiveInjector.resolveAndCreate(providers);
let compilerConfig: CompilerConfig = inj.get(CompilerConfig, null);
if (compilerConfig) {
// Note: forms read the platform directives / pipes, modify them
// and provide a CompilerConfig out of it
directives = directives.concat(compilerConfig.platformDirectives);
pipes = pipes.concat(compilerConfig.platformPipes);
} else {
// If nobody provided a CompilerConfig, use the
// PLATFORM_DIRECTIVES / PLATFORM_PIPES values directly.
directives = directives.concat(inj.get(PLATFORM_DIRECTIVES, []));
pipes = pipes.concat(inj.get(PLATFORM_PIPES, []));
}
let xhr = inj.get(XHR, null);
if (xhr) {
compilerProviders.push([{provide: XHR, useValue: xhr}]);
}
}
if (!compiler) {
compiler = browserCompiler({providers: compilerProviders});
} }
@AppModule({
providers: providers,
modules: modules.concat([BrowserModule]),
directives: directives,
pipes: pipes,
precompile: precompile.concat([appComponentType])
})
class DynamicModule {
constructor(public appRef: ApplicationRef) {}
}
return bootstrapModule(DynamicModule, compiler)
.then(
(moduleRef) => moduleRef.instance.appRef.waitForAsyncInitializers().then(
() => moduleRef.instance.appRef.bootstrap(appComponentType)));
}
/** /**
* @experimental * @experimental
@ -178,3 +306,7 @@ export function bootstrapWorkerApp(
workerAppPlatform().injector); workerAppPlatform().injector);
return coreLoadAndBootstrap(appComponentType, appInjector); return coreLoadAndBootstrap(appComponentType, appInjector);
} }
function normalizeArray(arr: any[]): any[] {
return arr ? arr : [];
}

View File

@ -9,3 +9,4 @@
import {__platform_browser_private__ as r, __platform_browser_private__ as t} from '@angular/platform-browser'; import {__platform_browser_private__ as r, __platform_browser_private__ as t} from '@angular/platform-browser';
export var getDOM: typeof t.getDOM = r.getDOM; export var getDOM: typeof t.getDOM = r.getDOM;
export var initDomAdapter: typeof t.initDomAdapter = r.initDomAdapter;

View File

@ -16,24 +16,12 @@ export var VIEW_ENCAPSULATION_VALUES: typeof t.VIEW_ENCAPSULATION_VALUES =
r.VIEW_ENCAPSULATION_VALUES; r.VIEW_ENCAPSULATION_VALUES;
export type DebugDomRootRenderer = t.DebugDomRootRenderer; export type DebugDomRootRenderer = t.DebugDomRootRenderer;
export var DebugDomRootRenderer: typeof t.DebugDomRootRenderer = r.DebugDomRootRenderer; export var DebugDomRootRenderer: typeof t.DebugDomRootRenderer = r.DebugDomRootRenderer;
/**
* @experimental bogus marker to pass the ts-api-guardian's check - this api should be public so
* this line will go away when that happens
*/
export var SecurityContext: typeof t.SecurityContext = r.SecurityContext;
export type SecurityContext = t.SecurityContext;
export var SanitizationService: typeof t.SanitizationService = r.SanitizationService;
export type SanitizationService = t.SanitizationService;
export var reflector: typeof t.reflector = r.reflector; export var reflector: typeof t.reflector = r.reflector;
export type NoOpAnimationPlayer = t.NoOpAnimationPlayer; export type NoOpAnimationPlayer = t.NoOpAnimationPlayer;
export var NoOpAnimationPlayer: typeof t.NoOpAnimationPlayer = r.NoOpAnimationPlayer; export var NoOpAnimationPlayer: typeof t.NoOpAnimationPlayer = r.NoOpAnimationPlayer;
export type AnimationPlayer = t.AnimationPlayer; export type AnimationPlayer = t.AnimationPlayer;
export var AnimationPlayer: typeof t.AnimationPlayer = r.AnimationPlayer; export var AnimationPlayer: typeof t.AnimationPlayer = r.AnimationPlayer;
export type NoOpAnimationDriver = t.NoOpAnimationDriver;
export var NoOpAnimationDriver: typeof t.NoOpAnimationDriver = r.NoOpAnimationDriver;
export type AnimationDriver = t.AnimationDriver;
export var AnimationDriver: typeof t.AnimationDriver = r.AnimationDriver;
export type AnimationSequencePlayer = t.AnimationSequencePlayer; export type AnimationSequencePlayer = t.AnimationSequencePlayer;
export var AnimationSequencePlayer: typeof t.AnimationSequencePlayer = r.AnimationSequencePlayer; export var AnimationSequencePlayer: typeof t.AnimationSequencePlayer = r.AnimationSequencePlayer;
export type AnimationGroupPlayer = t.AnimationGroupPlayer; export type AnimationGroupPlayer = t.AnimationGroupPlayer;

View File

@ -6,21 +6,21 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
export {BROWSER_APP_PROVIDERS, BROWSER_PLATFORM_PROVIDERS, BROWSER_SANITIZATION_PROVIDERS, BrowserModule, bootstrapModuleFactory, browserPlatform} from './src/browser';
export {BrowserPlatformLocation} from './src/browser/location/browser_platform_location'; export {BrowserPlatformLocation} from './src/browser/location/browser_platform_location';
export {Title} from './src/browser/title'; export {Title} from './src/browser/title';
export {disableDebugTools, enableDebugTools} from './src/browser/tools/tools'; export {disableDebugTools, enableDebugTools} from './src/browser/tools/tools';
export {AnimationDriver} from './src/dom/animation_driver';
export {By} from './src/dom/debug/by'; export {By} from './src/dom/debug/by';
export {DOCUMENT} from './src/dom/dom_tokens'; export {DOCUMENT} from './src/dom/dom_tokens';
export {EVENT_MANAGER_PLUGINS, EventManager} from './src/dom/events/event_manager'; export {EVENT_MANAGER_PLUGINS, EventManager} from './src/dom/events/event_manager';
export {HAMMER_GESTURE_CONFIG, HammerGestureConfig} from './src/dom/events/hammer_gestures'; export {HAMMER_GESTURE_CONFIG, HammerGestureConfig} from './src/dom/events/hammer_gestures';
export {DomSanitizationService, SafeHtml, SafeResourceUrl, SafeScript, SafeStyle, SafeUrl, SecurityContext} from './src/security/dom_sanitization_service'; export {DomSanitizationService, SafeHtml, SafeResourceUrl, SafeScript, SafeStyle, SafeUrl} from './src/security/dom_sanitization_service';
export * from './src/browser';
// Web Workers // Web Workers
export {ClientMessageBroker, ClientMessageBrokerFactory, FnArg, UiArguments} from './src/web_workers/shared/client_message_broker'; export {ClientMessageBroker, ClientMessageBrokerFactory, FnArg, UiArguments} from './src/web_workers/shared/client_message_broker';
export {ReceivedMessage, ServiceMessageBroker, ServiceMessageBrokerFactory} from './src/web_workers/shared/service_message_broker';
export {PRIMITIVE} from './src/web_workers/shared/serializer'; export {PRIMITIVE} from './src/web_workers/shared/serializer';
export {ReceivedMessage, ServiceMessageBroker, ServiceMessageBrokerFactory} from './src/web_workers/shared/service_message_broker';
export * from './src/web_workers/shared/message_bus'; export * from './src/web_workers/shared/message_bus';
export {WORKER_APP_LOCATION_PROVIDERS} from './src/web_workers/worker/location_providers'; export {WORKER_APP_LOCATION_PROVIDERS} from './src/web_workers/worker/location_providers';
export {WORKER_UI_LOCATION_PROVIDERS} from './src/web_workers/ui/location_providers'; export {WORKER_UI_LOCATION_PROVIDERS} from './src/web_workers/ui/location_providers';

View File

@ -6,6 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import * as browser from './src/browser';
import * as ng_proble from './src/dom/debug/ng_probe'; import * as ng_proble from './src/dom/debug/ng_probe';
import * as dom_adapter from './src/dom/dom_adapter'; import * as dom_adapter from './src/dom/dom_adapter';
import * as dom_renderer from './src/dom/dom_renderer'; import * as dom_renderer from './src/dom/dom_renderer';
@ -28,6 +29,7 @@ export declare namespace __platform_browser_private_types__ {
export var ELEMENT_PROBE_PROVIDERS: typeof ng_proble.ELEMENT_PROBE_PROVIDERS; export var ELEMENT_PROBE_PROVIDERS: typeof ng_proble.ELEMENT_PROBE_PROVIDERS;
export type DomEventsPlugin = dom_events.DomEventsPlugin; export type DomEventsPlugin = dom_events.DomEventsPlugin;
export var DomEventsPlugin: typeof dom_events.DomEventsPlugin; export var DomEventsPlugin: typeof dom_events.DomEventsPlugin;
export var initDomAdapter: typeof browser.initDomAdapter;
} }
export var __platform_browser_private__ = { export var __platform_browser_private__ = {
@ -39,5 +41,6 @@ export var __platform_browser_private__ = {
DomSharedStylesHost: shared_styles_host.DomSharedStylesHost, DomSharedStylesHost: shared_styles_host.DomSharedStylesHost,
SharedStylesHost: shared_styles_host.SharedStylesHost, SharedStylesHost: shared_styles_host.SharedStylesHost,
ELEMENT_PROBE_PROVIDERS: ng_proble.ELEMENT_PROBE_PROVIDERS, ELEMENT_PROBE_PROVIDERS: ng_proble.ELEMENT_PROBE_PROVIDERS,
DomEventsPlugin: dom_events.DomEventsPlugin DomEventsPlugin: dom_events.DomEventsPlugin,
initDomAdapter: browser.initDomAdapter
}; };

View File

@ -6,10 +6,11 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {FORM_PROVIDERS, PlatformLocation} from '@angular/common'; import {COMMON_DIRECTIVES, COMMON_PIPES, FORM_PROVIDERS, PlatformLocation} from '@angular/common';
import {APPLICATION_COMMON_PROVIDERS, ExceptionHandler, OpaqueToken, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, PlatformRef, ReflectiveInjector, RootRenderer, Testability, assertPlatform, createPlatform, getPlatform} from '@angular/core'; import {APPLICATION_COMMON_PROVIDERS, AppModule, AppModuleFactory, AppModuleRef, ExceptionHandler, NgZone, OpaqueToken, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, PlatformRef, ReflectiveInjector, RootRenderer, SanitizationService, Testability, assertPlatform, createPlatform, getPlatform, isDevMode} from '@angular/core';
import {AnimationDriver, NoOpAnimationDriver, SanitizationService, wtfInit} from '../core_private'; import {wtfInit} from '../core_private';
import {AnimationDriver} from '../src/dom/animation_driver';
import {WebAnimationsDriver} from '../src/dom/web_animations_driver'; import {WebAnimationsDriver} from '../src/dom/web_animations_driver';
import {BrowserDomAdapter} from './browser/browser_adapter'; import {BrowserDomAdapter} from './browser/browser_adapter';
@ -27,7 +28,6 @@ import {DomSharedStylesHost, SharedStylesHost} from './dom/shared_styles_host';
import {isBlank} from './facade/lang'; import {isBlank} from './facade/lang';
import {DomSanitizationService, DomSanitizationServiceImpl} from './security/dom_sanitization_service'; import {DomSanitizationService, DomSanitizationServiceImpl} from './security/dom_sanitization_service';
const BROWSER_PLATFORM_MARKER = new OpaqueToken('BrowserPlatformMarker'); const BROWSER_PLATFORM_MARKER = new OpaqueToken('BrowserPlatformMarker');
/** /**
@ -86,23 +86,71 @@ export function browserPlatform(): PlatformRef {
return assertPlatform(BROWSER_PLATFORM_MARKER); return assertPlatform(BROWSER_PLATFORM_MARKER);
} }
function initDomAdapter() { export function initDomAdapter() {
BrowserDomAdapter.makeCurrent(); BrowserDomAdapter.makeCurrent();
wtfInit(); wtfInit();
BrowserGetTestability.init(); BrowserGetTestability.init();
} }
function _exceptionHandler(): ExceptionHandler { export function _exceptionHandler(): ExceptionHandler {
return new ExceptionHandler(getDOM()); return new ExceptionHandler(getDOM());
} }
function _document(): any { export function _document(): any {
return getDOM().defaultDoc(); return getDOM().defaultDoc();
} }
function _resolveDefaultAnimationDriver(): AnimationDriver { export function _resolveDefaultAnimationDriver(): AnimationDriver {
if (getDOM().supportsWebAnimation()) { if (getDOM().supportsWebAnimation()) {
return new WebAnimationsDriver(); return new WebAnimationsDriver();
} }
return new NoOpAnimationDriver(); return AnimationDriver.NOOP;
}
/**
* The app module for the browser.
* @stable
*/
@AppModule({
providers: [
BROWSER_APP_PROVIDERS,
],
directives: COMMON_DIRECTIVES,
pipes: COMMON_PIPES
})
export class BrowserModule {
}
/**
* Creates an instance of an `@AppModule` for the browser platform
* for offline compilation.
*
* ## Simple Example
*
* ```typescript
* my_module.ts:
*
* @AppModule({
* modules: [BrowserModule]
* })
* class MyModule {}
*
* main.ts:
* import {MyModuleNgFactory} from './my_module.ngfactory';
* import {bootstrapModuleFactory} from '@angular/platform-browser';
*
* let moduleRef = bootstrapModuleFactory(MyModuleNgFactory);
* ```
* @stable
*/
export function bootstrapModuleFactory<M>(moduleFactory: AppModuleFactory<M>): AppModuleRef<M> {
let platformInjector = browserPlatform().injector;
// Note: We need to create the NgZone _before_ we instantiate the module,
// as instantiating the module creates some providers eagerly.
// So we create a mini parent injector that just contains the new NgZone and
// pass that as parent to the AppModuleFactory.
let ngZone = new NgZone({enableLongStackTrace: isDevMode()});
let ngZoneInjector =
ReflectiveInjector.resolveAndCreate([{provide: NgZone, useValue: ngZone}], platformInjector);
return ngZone.run(() => { return moduleFactory.create(ngZoneInjector); });
} }

View File

@ -6,20 +6,24 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {AnimationKeyframe} from './animation_keyframe'; import {AnimationPlayer} from '@angular/core';
import {AnimationPlayer, NoOpAnimationPlayer} from './animation_player';
import {AnimationStyles} from './animation_styles';
export abstract class AnimationDriver { import {AnimationKeyframe, AnimationStyles, NoOpAnimationPlayer} from '../../core_private';
abstract animate(
element: any, startingStyles: AnimationStyles, keyframes: AnimationKeyframe[],
duration: number, delay: number, easing: string): AnimationPlayer;
}
export class NoOpAnimationDriver extends AnimationDriver { class _NoOpAnimationDriver implements AnimationDriver {
animate( animate(
element: any, startingStyles: AnimationStyles, keyframes: AnimationKeyframe[], element: any, startingStyles: AnimationStyles, keyframes: AnimationKeyframe[],
duration: number, delay: number, easing: string): AnimationPlayer { duration: number, delay: number, easing: string): AnimationPlayer {
return new NoOpAnimationPlayer(); return new NoOpAnimationPlayer();
} }
} }
/**
* @experimental
*/
export abstract class AnimationDriver {
static NOOP: AnimationDriver = new _NoOpAnimationDriver();
abstract animate(
element: any, startingStyles: AnimationStyles, keyframes: AnimationKeyframe[],
duration: number, delay: number, easing: string): AnimationPlayer;
}

View File

@ -30,7 +30,7 @@ export function inspectNativeElement(element: any /** TODO #9100 */): DebugNode
return getDebugNode(element); return getDebugNode(element);
} }
function _createConditionalRootRenderer(rootRenderer: any /** TODO #9100 */) { export function _createConditionalRootRenderer(rootRenderer: any /** TODO #9100 */) {
if (isDevMode()) { if (isDevMode()) {
return _createRootRenderer(rootRenderer); return _createRootRenderer(rootRenderer);
} }

View File

@ -14,11 +14,12 @@ import {Json, RegExpWrapper, StringWrapper, isArray, isBlank, isPresent, isStrin
import {DomSharedStylesHost} from './shared_styles_host'; import {DomSharedStylesHost} from './shared_styles_host';
import {AnimationKeyframe, AnimationStyles, AnimationPlayer, AnimationDriver, RenderDebugInfo,} from '../../core_private'; import {AnimationKeyframe, AnimationStyles, AnimationPlayer, RenderDebugInfo,} from '../../core_private';
import {EventManager} from './events/event_manager'; import {EventManager} from './events/event_manager';
import {DOCUMENT} from './dom_tokens'; import {DOCUMENT} from './dom_tokens';
import {getDOM} from './dom_adapter'; import {getDOM} from './dom_adapter';
import {AnimationDriver} from './animation_driver';
import {camelCaseToDashCase} from './util'; import {camelCaseToDashCase} from './util';
const NAMESPACE_URIS = { const NAMESPACE_URIS = {

View File

@ -8,10 +8,11 @@
import {AUTO_STYLE, BaseException} from '@angular/core'; import {AUTO_STYLE, BaseException} from '@angular/core';
import {AnimationDriver, AnimationKeyframe, AnimationPlayer, AnimationStyles, NoOpAnimationPlayer} from '../../core_private'; import {AnimationKeyframe, AnimationPlayer, AnimationStyles, NoOpAnimationPlayer} from '../../core_private';
import {StringMapWrapper} from '../facade/collection'; import {StringMapWrapper} from '../facade/collection';
import {StringWrapper, isNumber, isPresent} from '../facade/lang'; import {StringWrapper, isNumber, isPresent} from '../facade/lang';
import {AnimationDriver} from './animation_driver';
import {getDOM} from './dom_adapter'; import {getDOM} from './dom_adapter';
import {DomAnimatePlayer} from './dom_animate_player'; import {DomAnimatePlayer} from './dom_animate_player';
import {dashCaseToCamelCase} from './util'; import {dashCaseToCamelCase} from './util';

View File

@ -6,9 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {Injectable} from '@angular/core'; import {Injectable, SanitizationService, SecurityContext} from '@angular/core';
import {SanitizationService, SecurityContext} from '../../core_private';
import {sanitizeHtml} from './html_sanitizer'; import {sanitizeHtml} from './html_sanitizer';
import {sanitizeStyle} from './style_sanitizer'; import {sanitizeStyle} from './style_sanitizer';

View File

@ -8,11 +8,12 @@
import {APPLICATION_COMMON_PROVIDERS, APP_INITIALIZER, ExceptionHandler, Injectable, Injector, NgZone, OpaqueToken, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, PlatformRef, ReflectiveInjector, RootRenderer, Testability, assertPlatform, createPlatform, getPlatform} from '@angular/core'; import {APPLICATION_COMMON_PROVIDERS, APP_INITIALIZER, ExceptionHandler, Injectable, Injector, NgZone, OpaqueToken, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, PlatformRef, ReflectiveInjector, RootRenderer, Testability, assertPlatform, createPlatform, getPlatform} from '@angular/core';
import {AnimationDriver, NoOpAnimationDriver, wtfInit} from '../core_private'; import {wtfInit} from '../core_private';
import {BROWSER_SANITIZATION_PROVIDERS} from './browser'; import {BROWSER_SANITIZATION_PROVIDERS} from './browser';
import {BrowserDomAdapter} from './browser/browser_adapter'; import {BrowserDomAdapter} from './browser/browser_adapter';
import {BrowserGetTestability} from './browser/testability'; import {BrowserGetTestability} from './browser/testability';
import {AnimationDriver} from './dom/animation_driver';
import {getDOM} from './dom/dom_adapter'; import {getDOM} from './dom/dom_adapter';
import {DomRootRenderer, DomRootRenderer_} from './dom/dom_renderer'; import {DomRootRenderer, DomRootRenderer_} from './dom/dom_renderer';
import {DOCUMENT} from './dom/dom_tokens'; import {DOCUMENT} from './dom/dom_tokens';
@ -178,5 +179,5 @@ function spawnWebWorker(uri: string, instance: WebWorkerInstance): void {
function _resolveDefaultAnimationDriver(): AnimationDriver { function _resolveDefaultAnimationDriver(): AnimationDriver {
// web workers have not been tested or configured to // web workers have not been tested or configured to
// work with animations just yet... // work with animations just yet...
return new NoOpAnimationDriver(); return AnimationDriver.NOOP;
} }

View File

@ -6,12 +6,15 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {APP_INITIALIZER, Component, Directive, ExceptionHandler, Inject, OnDestroy, PLATFORM_INITIALIZER, ReflectiveInjector, coreLoadAndBootstrap, createPlatform, provide} from '@angular/core'; import {LowerCasePipe, NgIf} from '@angular/common';
import {XHR} from '@angular/compiler';
import {APP_INITIALIZER, Component, Directive, ExceptionHandler, Inject, OnDestroy, PLATFORM_DIRECTIVES, PLATFORM_INITIALIZER, PLATFORM_PIPES, ReflectiveInjector, coreLoadAndBootstrap, createPlatform, provide} from '@angular/core';
import {ApplicationRef, disposePlatform} from '@angular/core/src/application_ref'; import {ApplicationRef, disposePlatform} from '@angular/core/src/application_ref';
import {Console} from '@angular/core/src/console'; import {Console} from '@angular/core/src/console';
import {ComponentRef} from '@angular/core/src/linker/component_factory'; import {ComponentRef} from '@angular/core/src/linker/component_factory';
import {Testability, TestabilityRegistry} from '@angular/core/src/testability/testability'; import {Testability, TestabilityRegistry} from '@angular/core/src/testability/testability';
import {AsyncTestCompleter, Log, afterEach, beforeEach, beforeEachProviders, describe, expect, inject, it} from '@angular/core/testing/testing_internal'; import {ComponentFixture} from '@angular/core/testing';
import {AsyncTestCompleter, Log, afterEach, beforeEach, beforeEachProviders, ddescribe, describe, expect, iit, inject, it} from '@angular/core/testing/testing_internal';
import {BROWSER_APP_PROVIDERS, BROWSER_PLATFORM_PROVIDERS} from '@angular/platform-browser'; import {BROWSER_APP_PROVIDERS, BROWSER_PLATFORM_PROVIDERS} from '@angular/platform-browser';
import {BROWSER_APP_COMPILER_PROVIDERS, bootstrap} from '@angular/platform-browser-dynamic'; import {BROWSER_APP_COMPILER_PROVIDERS, bootstrap} from '@angular/platform-browser-dynamic';
import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter'; import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter';
@ -69,6 +72,19 @@ class HelloOnDestroyTickCmp implements OnDestroy {
ngOnDestroy(): void { this.appRef.tick(); } ngOnDestroy(): void { this.appRef.tick(); }
} }
@Component({selector: 'hello-app', templateUrl: './sometemplate.html'})
class HelloUrlCmp {
greeting = 'hello';
}
@Component({
selector: 'hello-app',
template: `<div [title]="'HELLO' | lowercase"></div><div *ngIf="show"></div>`
})
class HelloCmpUsingPlatformDirectiveAndPipe {
show: boolean = false;
}
class _ArrayLogger { class _ArrayLogger {
res: any[] = []; res: any[] = [];
log(s: any): void { this.res.push(s); } log(s: any): void { this.res.push(s); }
@ -108,15 +124,9 @@ export function main() {
afterEach(disposePlatform); afterEach(disposePlatform);
it('should throw if bootstrapped Directive is not a Component', () => { it('should throw if bootstrapped Directive is not a Component', () => {
var logger = new _ArrayLogger(); expect(() => bootstrap(HelloRootDirectiveIsNotCmp))
var exceptionHandler = new ExceptionHandler(logger, false);
expect(
() => bootstrap(
HelloRootDirectiveIsNotCmp,
[testProviders, {provide: ExceptionHandler, useValue: exceptionHandler}]))
.toThrowError( .toThrowError(
`Could not compile '${stringify(HelloRootDirectiveIsNotCmp)}' because it is not a component.`); `Could not compile '${stringify(HelloRootDirectiveIsNotCmp)}' because it is not a component.`);
expect(logger.res.join('')).toContain('Could not compile');
}); });
it('should throw if no element is found', it('should throw if no element is found',
@ -270,5 +280,40 @@ export function main() {
}); });
}); });
})); }));
// Note: This will soon be deprecated as bootstrap creates a separate injector for the compiler,
// i.e. such providers needs to go into that injecotr (when calling `browserCompiler`);
it('should still allow to provide a custom xhr via the regular providers',
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
let spyXhr: XHR = {get: (url: string) => Promise.resolve('{{greeting}} world!')};
bootstrap(HelloUrlCmp, testProviders.concat([{provide: XHR, useValue: spyXhr}]))
.then((compRef) => {
expect(el).toHaveText('hello world!');
async.done();
});
}));
// Note: This will soon be deprecated as bootstrap creates a separate injector for the compiler,
// i.e. such providers needs to go into that injecotr (when calling `browserCompiler`);
it('should still allow to provide platform directives/pipes via the regular providers',
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
bootstrap(HelloCmpUsingPlatformDirectiveAndPipe, testProviders.concat([
{provide: PLATFORM_DIRECTIVES, useValue: [NgIf]},
{provide: PLATFORM_PIPES, useValue: [LowerCasePipe]}
])).then((compRef) => {
let compFixture = new ComponentFixture(compRef, null, null);
let el = compFixture.debugElement;
// Test that ngIf works
expect(el.children.length).toBe(1);
compFixture.componentInstance.show = true;
compFixture.detectChanges();
expect(el.children.length).toBe(2);
// Test that lowercase pipe works
expect(el.children[0].properties['title']).toBe('hello');
async.done();
});
}));
}); });
} }

View File

@ -9,14 +9,15 @@
import {LocationStrategy} from '@angular/common'; import {LocationStrategy} from '@angular/common';
import {APP_ID, NgZone, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER} from '@angular/core'; import {APP_ID, NgZone, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER} from '@angular/core';
import {AnimationDriver, NoOpAnimationDriver} from '../core_private';
import {BROWSER_APP_PROVIDERS} from '../src/browser'; import {BROWSER_APP_PROVIDERS} from '../src/browser';
import {BrowserDomAdapter} from '../src/browser/browser_adapter'; import {BrowserDomAdapter} from '../src/browser/browser_adapter';
import {AnimationDriver} from '../src/dom/animation_driver';
import {ELEMENT_PROBE_PROVIDERS} from '../src/dom/debug/ng_probe'; import {ELEMENT_PROBE_PROVIDERS} from '../src/dom/debug/ng_probe';
import {BrowserDetection} from './browser_util'; import {BrowserDetection} from './browser_util';
/** /**
* Default platform providers for testing without a compiler. * Default platform providers for testing without a compiler.
*/ */
@ -28,7 +29,7 @@ const TEST_BROWSER_STATIC_PLATFORM_PROVIDERS: Array<any /*Type | Provider | any[
const ADDITIONAL_TEST_BROWSER_STATIC_PROVIDERS: Array<any /*Type | Provider | any[]*/> = [ const ADDITIONAL_TEST_BROWSER_STATIC_PROVIDERS: Array<any /*Type | Provider | any[]*/> = [
{provide: APP_ID, useValue: 'a'}, ELEMENT_PROBE_PROVIDERS, {provide: APP_ID, useValue: 'a'}, ELEMENT_PROBE_PROVIDERS,
{provide: NgZone, useFactory: createNgZone}, {provide: NgZone, useFactory: createNgZone},
{provide: AnimationDriver, useClass: NoOpAnimationDriver} {provide: AnimationDriver, useValue: AnimationDriver.NOOP}
]; ];

View File

@ -6,12 +6,12 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {AnimationDriver} from '../../src/animation/animation_driver'; import {AnimationPlayer} from '@angular/core';
import {AnimationKeyframe} from '../../src/animation/animation_keyframe'; import {MockAnimationPlayer} from '@angular/core/testing/testing_internal';
import {AnimationPlayer} from '../../src/animation/animation_player';
import {AnimationStyles} from '../../src/animation/animation_styles'; import {AnimationKeyframe, AnimationStyles} from '../core_private';
import {StringMapWrapper} from '../../src/facade/collection'; import {AnimationDriver} from '../src/dom/animation_driver';
import {MockAnimationPlayer} from '../../testing/animation/mock_animation_player'; import {StringMapWrapper} from '../src/facade/collection';
export class MockAnimationDriver extends AnimationDriver { export class MockAnimationDriver extends AnimationDriver {
log: any[] /** TODO #9100 */ = []; log: any[] /** TODO #9100 */ = [];

View File

@ -11,7 +11,3 @@ import {__core_private__ as r, __core_private_types__ as t} from '@angular/core'
export var reflector: typeof t.reflector = r.reflector; export var reflector: typeof t.reflector = r.reflector;
export var ReflectionCapabilities: typeof t.ReflectionCapabilities = r.ReflectionCapabilities; export var ReflectionCapabilities: typeof t.ReflectionCapabilities = r.ReflectionCapabilities;
export var wtfInit: typeof t.wtfInit = r.wtfInit; export var wtfInit: typeof t.wtfInit = r.wtfInit;
export type NoOpAnimationDriver = t.NoOpAnimationDriver;
export var NoOpAnimationDriver: typeof t.NoOpAnimationDriver = r.NoOpAnimationDriver;
export type AnimationDriver = t.AnimationDriver;
export var AnimationDriver: typeof t.AnimationDriver = r.AnimationDriver;

View File

@ -10,9 +10,8 @@ import {COMPILER_PROVIDERS, DirectiveResolver, ViewResolver, XHR} from '@angular
import {MockDirectiveResolver, MockViewResolver, OverridingTestComponentBuilder} from '@angular/compiler/testing'; import {MockDirectiveResolver, MockViewResolver, OverridingTestComponentBuilder} from '@angular/compiler/testing';
import {APPLICATION_COMMON_PROVIDERS, APP_ID, NgZone, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, RootRenderer} from '@angular/core'; import {APPLICATION_COMMON_PROVIDERS, APP_ID, NgZone, PLATFORM_COMMON_PROVIDERS, PLATFORM_INITIALIZER, RootRenderer} from '@angular/core';
import {TestComponentBuilder, TestComponentRenderer} from '@angular/core/testing'; import {TestComponentBuilder, TestComponentRenderer} from '@angular/core/testing';
import {BROWSER_SANITIZATION_PROVIDERS, DOCUMENT, EVENT_MANAGER_PLUGINS, EventManager} from '@angular/platform-browser'; import {AnimationDriver, BROWSER_SANITIZATION_PROVIDERS, DOCUMENT, EVENT_MANAGER_PLUGINS, EventManager} from '@angular/platform-browser';
import {AnimationDriver, NoOpAnimationDriver} from '../core_private';
import {DOMTestComponentRenderer} from '../platform_browser_dynamic_testing_private'; import {DOMTestComponentRenderer} from '../platform_browser_dynamic_testing_private';
import {DomEventsPlugin, DomRootRenderer, DomRootRenderer_, DomSharedStylesHost, ELEMENT_PROBE_PROVIDERS, SharedStylesHost, getDOM} from '../platform_browser_private'; import {DomEventsPlugin, DomRootRenderer, DomRootRenderer_, DomSharedStylesHost, ELEMENT_PROBE_PROVIDERS, SharedStylesHost, getDOM} from '../platform_browser_private';
import {Parse5DomAdapter} from '../src/parse5_adapter'; import {Parse5DomAdapter} from '../src/parse5_adapter';
@ -63,7 +62,7 @@ export const TEST_SERVER_APPLICATION_PROVIDERS: Array<any /*Type | Provider | an
/* @ts2dart_Provider */ {provide: DOCUMENT, useFactory: appDoc}, /* @ts2dart_Provider */ {provide: DOCUMENT, useFactory: appDoc},
/* @ts2dart_Provider */ {provide: DomRootRenderer, useClass: DomRootRenderer_}, /* @ts2dart_Provider */ {provide: DomRootRenderer, useClass: DomRootRenderer_},
/* @ts2dart_Provider */ {provide: RootRenderer, useExisting: DomRootRenderer}, /* @ts2dart_Provider */ {provide: RootRenderer, useExisting: DomRootRenderer},
/* @ts2dart_Provider */ {provide: AnimationDriver, useClass: NoOpAnimationDriver}, /* @ts2dart_Provider */ {provide: AnimationDriver, useValue: AnimationDriver.NOOP},
EventManager, EventManager,
/* @ts2dart_Provider */ { /* @ts2dart_Provider */ {
provide: EVENT_MANAGER_PLUGINS, provide: EVENT_MANAGER_PLUGINS,

View File

@ -0,0 +1,6 @@
# How to run the examples locally
$ cp -r ./modules/playground ./dist/all/
$ tsc -p modules --emitDecoratorMetadata -w
$ gulp serve
$ open http://localhost:8000/all/playground/src/hello_world/index.html?bundles=false

View File

@ -50,12 +50,10 @@ declare var System: any;
System.config({ System.config({
map: {'index': 'index.js', '@angular': '/all/@angular'}, map: {'index': 'index.js', '@angular': '/all/@angular'},
packages: { packages: {
// 'app': { 'app': { defaultExtension: 'js' },
// main: 'index.js',
// defaultExtension: 'js'
// },
'@angular/core': {main: 'index.js', defaultExtension: 'js'}, '@angular/core': {main: 'index.js', defaultExtension: 'js'},
'@angular/compiler': {main: 'index.js', defaultExtension: 'js'}, '@angular/compiler': {main: 'index.js', defaultExtension: 'js'},
'@angular/router': {main: 'index.js', defaultExtension: 'js'},
'@angular/common': {main: 'index.js', defaultExtension: 'js'}, '@angular/common': {main: 'index.js', defaultExtension: 'js'},
'@angular/platform-browser': {main: 'index.js', defaultExtension: 'js'}, '@angular/platform-browser': {main: 'index.js', defaultExtension: 'js'},
'@angular/platform-browser-dynamic': {main: 'index.js', defaultExtension: 'js'} '@angular/platform-browser-dynamic': {main: 'index.js', defaultExtension: 'js'}

View File

@ -129,7 +129,7 @@ export declare abstract class ApplicationRef {
componentTypes: Type[]; componentTypes: Type[];
injector: Injector; injector: Injector;
zone: NgZone; zone: NgZone;
abstract bootstrap<C>(componentFactory: ComponentFactory<C>): ComponentRef<C>; abstract bootstrap<C>(componentFactory: ComponentFactory<C> | ConcreteType<C>): ComponentRef<C>;
abstract dispose(): void; abstract dispose(): void;
abstract registerBootstrapListener(listener: (ref: ComponentRef<any>) => void): void; abstract registerBootstrapListener(listener: (ref: ComponentRef<any>) => void): void;
abstract registerDisposeListener(dispose: () => void): void; abstract registerDisposeListener(dispose: () => void): void;
@ -289,14 +289,8 @@ export declare class Compiler {
clearCacheFor(type: Type): void; clearCacheFor(type: Type): void;
compileAppModuleAsync<T>(moduleType: ConcreteType<T>, metadata?: AppModuleMetadata): Promise<AppModuleFactory<T>>; compileAppModuleAsync<T>(moduleType: ConcreteType<T>, metadata?: AppModuleMetadata): Promise<AppModuleFactory<T>>;
compileAppModuleSync<T>(moduleType: ConcreteType<T>, metadata?: AppModuleMetadata): AppModuleFactory<T>; compileAppModuleSync<T>(moduleType: ConcreteType<T>, metadata?: AppModuleMetadata): AppModuleFactory<T>;
compileComponentAsync<T>(component: ConcreteType<T>, {moduleDirectives, modulePipes}?: { compileComponentAsync<T>(component: ConcreteType<T>): Promise<ComponentFactory<T>>;
moduleDirectives?: ConcreteType<any>[]; compileComponentSync<T>(component: ConcreteType<T>): ComponentFactory<T>;
modulePipes?: ConcreteType<any>[];
}): Promise<ComponentFactory<T>>;
compileComponentSync<T>(component: ConcreteType<T>, {moduleDirectives, modulePipes}?: {
moduleDirectives?: ConcreteType<any>[];
modulePipes?: ConcreteType<any>[];
}): ComponentFactory<T>;
} }
/** @stable */ /** @stable */
@ -501,9 +495,6 @@ export declare function coreBootstrap<C>(componentFactory: ComponentFactory<C>,
/** @experimental */ /** @experimental */
export declare function coreLoadAndBootstrap(componentType: Type, injector: Injector): Promise<ComponentRef<any>>; export declare function coreLoadAndBootstrap(componentType: Type, injector: Injector): Promise<ComponentRef<any>>;
/** @experimental */
export declare function createNgZone(): NgZone;
/** @experimental */ /** @experimental */
export declare function createPlatform(injector: Injector): PlatformRef; export declare function createPlatform(injector: Injector): PlatformRef;
@ -1224,6 +1215,21 @@ export declare abstract class RootRenderer {
abstract renderComponent(componentType: RenderComponentType): Renderer; abstract renderComponent(componentType: RenderComponentType): Renderer;
} }
/** @stable */
export declare abstract class SanitizationService {
abstract sanitize(context: SecurityContext, value: string): string;
}
/** @stable */
export declare enum SecurityContext {
NONE = 0,
HTML = 1,
STYLE = 2,
SCRIPT = 3,
URL = 4,
RESOURCE_URL = 5,
}
/** @stable */ /** @stable */
export declare var Self: SelfMetadataFactory; export declare var Self: SelfMetadataFactory;

View File

@ -16,7 +16,7 @@ export declare function beforeEachProviders(fn: () => Array<any>): void;
/** @stable */ /** @stable */
export declare class ComponentFixture<T> { export declare class ComponentFixture<T> {
changeDetectorRef: ChangeDetectorRef; changeDetectorRef: ChangeDetectorRef;
componentInstance: any; componentInstance: T;
componentRef: ComponentRef<T>; componentRef: ComponentRef<T>;
debugElement: DebugElement; debugElement: DebugElement;
elementRef: ElementRef; elementRef: ElementRef;

View File

@ -1,5 +1,8 @@
/** @experimental */ /** @experimental */
export declare function bootstrap(appComponentType: Type, customProviders?: Array<any>): Promise<ComponentRef<any>>; export declare function bootstrap<C>(appComponentType: ConcreteType<C>, customProviders?: Array<any>): Promise<ComponentRef<C>>;
/** @stable */
export declare function bootstrapModule<M>(moduleType: ConcreteType<M>, compiler?: Compiler): Promise<AppModuleRef<M>>;
/** @experimental */ /** @experimental */
export declare function bootstrapWorkerApp(appComponentType: Type, customProviders?: Array<any>): Promise<ComponentRef<any>>; export declare function bootstrapWorkerApp(appComponentType: Type, customProviders?: Array<any>): Promise<ComponentRef<any>>;
@ -10,5 +13,12 @@ export declare function bootstrapWorkerUi(workerScriptUri: string, customProvide
/** @experimental */ /** @experimental */
export declare const BROWSER_APP_COMPILER_PROVIDERS: Array<any>; export declare const BROWSER_APP_COMPILER_PROVIDERS: Array<any>;
/** @stable */
export declare function browserCompiler({useDebug, useJit, providers}?: {
useDebug?: boolean;
useJit?: boolean;
providers?: Array<any>;
}): Compiler;
/** @experimental */ /** @experimental */
export declare const CACHED_TEMPLATE_PROVIDER: Array<any>; export declare const CACHED_TEMPLATE_PROVIDER: Array<any>;

View File

@ -1,3 +1,12 @@
/** @experimental */
export declare abstract class AnimationDriver {
abstract animate(element: any, startingStyles: AnimationStyles, keyframes: AnimationKeyframe[], duration: number, delay: number, easing: string): AnimationPlayer;
static NOOP: AnimationDriver;
}
/** @stable */
export declare function bootstrapModuleFactory<M>(moduleFactory: AppModuleFactory<M>): AppModuleRef<M>;
/** @experimental */ /** @experimental */
export declare const BROWSER_APP_PROVIDERS: Array<any>; export declare const BROWSER_APP_PROVIDERS: Array<any>;
@ -7,6 +16,10 @@ export declare const BROWSER_PLATFORM_PROVIDERS: Array<any>;
/** @experimental */ /** @experimental */
export declare const BROWSER_SANITIZATION_PROVIDERS: Array<any>; export declare const BROWSER_SANITIZATION_PROVIDERS: Array<any>;
/** @stable */
export declare class BrowserModule {
}
/** @experimental */ /** @experimental */
export declare function browserPlatform(): PlatformRef; export declare function browserPlatform(): PlatformRef;
@ -147,9 +160,6 @@ export interface SafeStyle extends SafeValue {
export interface SafeUrl extends SafeValue { export interface SafeUrl extends SafeValue {
} }
/** @experimental */
export declare var SecurityContext: typeof t.SecurityContext;
/** @experimental */ /** @experimental */
export declare abstract class ServiceMessageBroker { export declare abstract class ServiceMessageBroker {
abstract registerMethod(methodName: string, signature: Type[], method: Function, returnType?: Type): void; abstract registerMethod(methodName: string, signature: Type[], method: Function, returnType?: Type): void;