feat(NgComponentOutlet): add NgModule support to NgComponentOutlet directive (#14088)
Allow NgComponentOutlet to dynamically load a module, then load a component from that module. Useful for lazy loading code, then add the lazy loaded code to the page using NgComponentOutlet. Closes #14043
This commit is contained in:
parent
4106d18172
commit
3ef73c2b19
|
@ -6,7 +6,8 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {ComponentFactoryResolver, ComponentRef, Directive, Injector, Input, OnChanges, Provider, SimpleChanges, Type, ViewContainerRef} from '@angular/core';
|
import {ComponentFactoryResolver, ComponentRef, Directive, Injector, Input, NgModuleFactory, NgModuleRef, OnChanges, OnDestroy, Provider, SimpleChanges, Type, ViewContainerRef} from '@angular/core';
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -20,16 +21,17 @@ import {ComponentFactoryResolver, ComponentRef, Directive, Injector, Input, OnCh
|
||||||
*
|
*
|
||||||
* You can control the component creation process by using the following optional attributes:
|
* You can control the component creation process by using the following optional attributes:
|
||||||
*
|
*
|
||||||
* * `ngOutletInjector`: Optional custom {@link Injector} that will be used as parent for the
|
* * `ngComponentOutletInjector`: Optional custom {@link Injector} that will be used as parent for
|
||||||
* Component.
|
* the Component. Defaults to the injector of the current view container.
|
||||||
* Defaults to the injector of the current view container.
|
|
||||||
*
|
*
|
||||||
* * `ngOutletProviders`: Optional injectable objects ({@link Provider}) that are visible to the
|
* * `ngComponentOutletProviders`: Optional injectable objects ({@link Provider}) that are visible
|
||||||
* component.
|
* to the component.
|
||||||
*
|
*
|
||||||
* * `ngOutletContent`: Optional list of projectable nodes to insert into the content
|
* * `ngComponentOutletContent`: Optional list of projectable nodes to insert into the content
|
||||||
* section of the component, if exists. ({@link NgContent}).
|
* section of the component, if exists.
|
||||||
*
|
*
|
||||||
|
* * `ngComponentOutletNgModuleFactory`: Optional module factory to allow dynamically loading other
|
||||||
|
* module, then load a component from that module.
|
||||||
*
|
*
|
||||||
* ### Syntax
|
* ### Syntax
|
||||||
*
|
*
|
||||||
|
@ -38,14 +40,20 @@ import {ComponentFactoryResolver, ComponentRef, Directive, Injector, Input, OnCh
|
||||||
* <ng-container *ngComponentOutlet="componentTypeExpression"></ng-container>
|
* <ng-container *ngComponentOutlet="componentTypeExpression"></ng-container>
|
||||||
* ```
|
* ```
|
||||||
*
|
*
|
||||||
* Customized
|
* Customized injector/content
|
||||||
* ```
|
* ```
|
||||||
* <ng-container *ngComponentOutlet="componentTypeExpression;
|
* <ng-container *ngComponentOutlet="componentTypeExpression;
|
||||||
* injector: injectorExpression;
|
* injector: injectorExpression;
|
||||||
* content: contentNodesExpression">
|
* content: contentNodesExpression;">
|
||||||
* </ng-container>
|
* </ng-container>
|
||||||
* ```
|
* ```
|
||||||
*
|
*
|
||||||
|
* Customized ngModuleFactory
|
||||||
|
* ```
|
||||||
|
* <ng-container *ngComponentOutlet="componentTypeExpression;
|
||||||
|
* ngModuleFactory: moduleFactory;">
|
||||||
|
* </ng-container>
|
||||||
|
* ```
|
||||||
* # Example
|
* # Example
|
||||||
*
|
*
|
||||||
* {@example common/ngComponentOutlet/ts/module.ts region='SimpleExample'}
|
* {@example common/ngComponentOutlet/ts/module.ts region='SimpleExample'}
|
||||||
|
@ -53,34 +61,55 @@ import {ComponentFactoryResolver, ComponentRef, Directive, Injector, Input, OnCh
|
||||||
* A more complete example with additional options:
|
* A more complete example with additional options:
|
||||||
*
|
*
|
||||||
* {@example common/ngComponentOutlet/ts/module.ts region='CompleteExample'}
|
* {@example common/ngComponentOutlet/ts/module.ts region='CompleteExample'}
|
||||||
|
|
||||||
|
* A more complete example with ngModuleFactory:
|
||||||
|
*
|
||||||
|
* {@example common/ngComponentOutlet/ts/module.ts region='NgModuleFactoryExample'}
|
||||||
*
|
*
|
||||||
* @experimental
|
* @experimental
|
||||||
*/
|
*/
|
||||||
@Directive({selector: '[ngComponentOutlet]'})
|
@Directive({selector: '[ngComponentOutlet]'})
|
||||||
export class NgComponentOutlet implements OnChanges {
|
export class NgComponentOutlet implements OnChanges, OnDestroy {
|
||||||
@Input() ngComponentOutlet: Type<any>;
|
@Input() ngComponentOutlet: Type<any>;
|
||||||
@Input() ngComponentOutletInjector: Injector;
|
@Input() ngComponentOutletInjector: Injector;
|
||||||
@Input() ngComponentOutletContent: any[][];
|
@Input() ngComponentOutletContent: any[][];
|
||||||
|
@Input() ngComponentOutletNgModuleFactory: NgModuleFactory<any>;
|
||||||
|
|
||||||
componentRef: ComponentRef<any>;
|
private _componentRef: ComponentRef<any> = null;
|
||||||
|
private _moduleRef: NgModuleRef<any> = null;
|
||||||
|
|
||||||
constructor(
|
constructor(private _viewContainerRef: ViewContainerRef) {}
|
||||||
private _cmpFactoryResolver: ComponentFactoryResolver,
|
|
||||||
private _viewContainerRef: ViewContainerRef) {}
|
|
||||||
|
|
||||||
ngOnChanges(changes: SimpleChanges) {
|
ngOnChanges(changes: SimpleChanges) {
|
||||||
if (this.componentRef) {
|
if (this._componentRef) {
|
||||||
this._viewContainerRef.remove(this._viewContainerRef.indexOf(this.componentRef.hostView));
|
this._viewContainerRef.remove(this._viewContainerRef.indexOf(this._componentRef.hostView));
|
||||||
}
|
}
|
||||||
this._viewContainerRef.clear();
|
this._viewContainerRef.clear();
|
||||||
this.componentRef = null;
|
this._componentRef = null;
|
||||||
|
|
||||||
if (this.ngComponentOutlet) {
|
if (this.ngComponentOutlet) {
|
||||||
let injector = this.ngComponentOutletInjector || this._viewContainerRef.parentInjector;
|
let injector = this.ngComponentOutletInjector || this._viewContainerRef.parentInjector;
|
||||||
|
|
||||||
this.componentRef = this._viewContainerRef.createComponent(
|
if ((changes as any).ngComponentOutletNgModuleFactory) {
|
||||||
this._cmpFactoryResolver.resolveComponentFactory(this.ngComponentOutlet),
|
if (this._moduleRef) this._moduleRef.destroy();
|
||||||
this._viewContainerRef.length, injector, this.ngComponentOutletContent);
|
if (this.ngComponentOutletNgModuleFactory) {
|
||||||
|
this._moduleRef = this.ngComponentOutletNgModuleFactory.create(injector);
|
||||||
|
} else {
|
||||||
|
this._moduleRef = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this._moduleRef) {
|
||||||
|
injector = this._moduleRef.injector;
|
||||||
|
}
|
||||||
|
|
||||||
|
let componentFactory =
|
||||||
|
injector.get(ComponentFactoryResolver).resolveComponentFactory(this.ngComponentOutlet);
|
||||||
|
|
||||||
|
this._componentRef = this._viewContainerRef.createComponent(
|
||||||
|
componentFactory, this._viewContainerRef.length, injector, this.ngComponentOutletContent);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
ngOnDestroy() {
|
||||||
|
if (this._moduleRef) this._moduleRef.destroy();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,8 +8,8 @@
|
||||||
|
|
||||||
import {CommonModule} from '@angular/common';
|
import {CommonModule} from '@angular/common';
|
||||||
import {NgComponentOutlet} from '@angular/common/src/directives/ng_component_outlet';
|
import {NgComponentOutlet} from '@angular/common/src/directives/ng_component_outlet';
|
||||||
import {Component, ComponentRef, Inject, InjectionToken, Injector, NO_ERRORS_SCHEMA, NgModule, Optional, Provider, QueryList, ReflectiveInjector, TemplateRef, Type, ViewChild, ViewChildren, ViewContainerRef} from '@angular/core';
|
import {Compiler, Component, ComponentRef, Inject, InjectionToken, Injector, NO_ERRORS_SCHEMA, NgModule, NgModuleFactory, Optional, Provider, QueryList, ReflectiveInjector, TemplateRef, Type, ViewChild, ViewChildren, ViewContainerRef} from '@angular/core';
|
||||||
import {TestBed, async} from '@angular/core/testing';
|
import {TestBed, async, fakeAsync} from '@angular/core/testing';
|
||||||
import {expect} from '@angular/platform-browser/testing/matchers';
|
import {expect} from '@angular/platform-browser/testing/matchers';
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
|
@ -143,6 +143,69 @@ export function main() {
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
expect(fixture.nativeElement).toHaveText('projected foo');
|
expect(fixture.nativeElement).toHaveText('projected foo');
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
it('should resolve components from other modules, if supplied', async(() => {
|
||||||
|
const compiler = TestBed.get(Compiler) as Compiler;
|
||||||
|
let fixture = TestBed.createComponent(TestComponent);
|
||||||
|
|
||||||
|
fixture.detectChanges();
|
||||||
|
expect(fixture.nativeElement).toHaveText('');
|
||||||
|
|
||||||
|
fixture.componentInstance.module = compiler.compileModuleSync(TestModule2);
|
||||||
|
fixture.componentInstance.currentComponent = Module2InjectedComponent;
|
||||||
|
|
||||||
|
fixture.detectChanges();
|
||||||
|
expect(fixture.nativeElement).toHaveText('baz');
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should clean up moduleRef, if supplied', async(() => {
|
||||||
|
let destroyed = false;
|
||||||
|
const compiler = TestBed.get(Compiler) as Compiler;
|
||||||
|
const fixture = TestBed.createComponent(TestComponent);
|
||||||
|
fixture.componentInstance.module = compiler.compileModuleSync(TestModule2);
|
||||||
|
fixture.componentInstance.currentComponent = Module2InjectedComponent;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
const moduleRef = fixture.componentInstance.ngComponentOutlet['_moduleRef'];
|
||||||
|
spyOn(moduleRef, 'destroy').and.callThrough();
|
||||||
|
|
||||||
|
expect(moduleRef.destroy).not.toHaveBeenCalled();
|
||||||
|
fixture.destroy();
|
||||||
|
expect(moduleRef.destroy).toHaveBeenCalled();
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should not re-create moduleRef when it didn\'t actually change', async(() => {
|
||||||
|
const compiler = TestBed.get(Compiler) as Compiler;
|
||||||
|
const fixture = TestBed.createComponent(TestComponent);
|
||||||
|
fixture.componentInstance.module = compiler.compileModuleSync(TestModule2);
|
||||||
|
fixture.componentInstance.currentComponent = Module2InjectedComponent;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('baz');
|
||||||
|
|
||||||
|
const moduleRef = fixture.componentInstance.ngComponentOutlet['_moduleRef'];
|
||||||
|
fixture.componentInstance.currentComponent = Module2InjectedComponent2;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('baz2');
|
||||||
|
expect(moduleRef).toBe(fixture.componentInstance.ngComponentOutlet['_moduleRef']);
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should re-create moduleRef when changed', async(() => {
|
||||||
|
const compiler = TestBed.get(Compiler) as Compiler;
|
||||||
|
const fixture = TestBed.createComponent(TestComponent);
|
||||||
|
fixture.componentInstance.module = compiler.compileModuleSync(TestModule2);
|
||||||
|
fixture.componentInstance.currentComponent = Module2InjectedComponent;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('baz');
|
||||||
|
|
||||||
|
fixture.componentInstance.module = compiler.compileModuleSync(TestModule3);
|
||||||
|
fixture.componentInstance.currentComponent = Module3InjectedComponent;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('bat');
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -158,15 +221,16 @@ class InjectedComponentAgain {
|
||||||
}
|
}
|
||||||
|
|
||||||
const TEST_CMP_TEMPLATE =
|
const TEST_CMP_TEMPLATE =
|
||||||
`<template *ngComponentOutlet="currentComponent; injector: injector; content: projectables"></template>`;
|
`<template *ngComponentOutlet="currentComponent; injector: injector; content: projectables; ngModuleFactory: module;"></template>`;
|
||||||
@Component({selector: 'test-cmp', template: TEST_CMP_TEMPLATE})
|
@Component({selector: 'test-cmp', template: TEST_CMP_TEMPLATE})
|
||||||
class TestComponent {
|
class TestComponent {
|
||||||
currentComponent: Type<any>;
|
currentComponent: Type<any>;
|
||||||
injector: Injector;
|
injector: Injector;
|
||||||
projectables: any[][];
|
projectables: any[][];
|
||||||
|
module: NgModuleFactory<any>;
|
||||||
|
|
||||||
get cmpRef(): ComponentRef<any> { return this.ngComponentOutlet.componentRef; }
|
get cmpRef(): ComponentRef<any> { return this.ngComponentOutlet['_componentRef']; }
|
||||||
set cmpRef(value: ComponentRef<any>) { this.ngComponentOutlet.componentRef = value; }
|
set cmpRef(value: ComponentRef<any>) { this.ngComponentOutlet['_componentRef'] = value; }
|
||||||
|
|
||||||
@ViewChildren(TemplateRef) tplRefs: QueryList<TemplateRef<any>>;
|
@ViewChildren(TemplateRef) tplRefs: QueryList<TemplateRef<any>>;
|
||||||
@ViewChild(NgComponentOutlet) ngComponentOutlet: NgComponentOutlet;
|
@ViewChild(NgComponentOutlet) ngComponentOutlet: NgComponentOutlet;
|
||||||
|
@ -182,3 +246,33 @@ class TestComponent {
|
||||||
})
|
})
|
||||||
export class TestModule {
|
export class TestModule {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Component({selector: 'mdoule-2-injected-component', template: 'baz'})
|
||||||
|
class Module2InjectedComponent {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({selector: 'mdoule-2-injected-component-2', template: 'baz2'})
|
||||||
|
class Module2InjectedComponent2 {
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
imports: [CommonModule],
|
||||||
|
declarations: [Module2InjectedComponent, Module2InjectedComponent2],
|
||||||
|
exports: [Module2InjectedComponent, Module2InjectedComponent2],
|
||||||
|
entryComponents: [Module2InjectedComponent, Module2InjectedComponent2]
|
||||||
|
})
|
||||||
|
export class TestModule2 {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({selector: 'mdoule-3-injected-component', template: 'bat'})
|
||||||
|
class Module3InjectedComponent {
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
imports: [CommonModule],
|
||||||
|
declarations: [Module3InjectedComponent],
|
||||||
|
exports: [Module3InjectedComponent],
|
||||||
|
entryComponents: [Module3InjectedComponent]
|
||||||
|
})
|
||||||
|
export class TestModule3 {
|
||||||
|
}
|
|
@ -31,5 +31,13 @@ describe('ngComponentOutlet', () => {
|
||||||
waitForElement('ng-component-outlet-complete-example');
|
waitForElement('ng-component-outlet-complete-example');
|
||||||
expect(element.all(by.css('complete-component')).getText()).toEqual(['Complete: Ahoj Svet!']);
|
expect(element.all(by.css('complete-component')).getText()).toEqual(['Complete: Ahoj Svet!']);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should render other module', () => {
|
||||||
|
browser.get(URL);
|
||||||
|
waitForElement('ng-component-outlet-other-module-example');
|
||||||
|
expect(element.all(by.css('other-module-component')).getText()).toEqual([
|
||||||
|
'Other Module Component!'
|
||||||
|
]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -6,7 +6,8 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Component, Injectable, Injector, NgModule, ReflectiveInjector} from '@angular/core';
|
import {CommonModule} from '@angular/common';
|
||||||
|
import {Compiler, Component, Injectable, Injector, NgModule, NgModuleFactory, ReflectiveInjector} from '@angular/core';
|
||||||
import {BrowserModule} from '@angular/platform-browser';
|
import {BrowserModule} from '@angular/platform-browser';
|
||||||
|
|
||||||
|
|
||||||
|
@ -60,12 +61,34 @@ class NgTemplateOutletCompleteExample {
|
||||||
}
|
}
|
||||||
// #enddocregion
|
// #enddocregion
|
||||||
|
|
||||||
|
// #docregion NgModuleFactoryExample
|
||||||
|
@Component({selector: 'other-module-component', template: `Other Module Component!`})
|
||||||
|
class OtherModuleComponent {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'ng-component-outlet-other-module-example',
|
||||||
|
template: `
|
||||||
|
<ng-container *ngComponentOutlet="OtherModuleComponent;
|
||||||
|
ngModuleFactory: myModule;"></ng-container>`
|
||||||
|
})
|
||||||
|
class NgTemplateOutletOtherModuleExample {
|
||||||
|
// This field is necessary to expose OtherModuleComponent to the template.
|
||||||
|
OtherModuleComponent = OtherModuleComponent;
|
||||||
|
myModule: NgModuleFactory<any>;
|
||||||
|
|
||||||
|
constructor(compiler: Compiler) { this.myModule = compiler.compileModuleSync(OtherModule); }
|
||||||
|
}
|
||||||
|
// #enddocregion
|
||||||
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'example-app',
|
selector: 'example-app',
|
||||||
template: `<ng-component-outlet-simple-example></ng-component-outlet-simple-example>
|
template: `<ng-component-outlet-simple-example></ng-component-outlet-simple-example>
|
||||||
<hr/>
|
<hr/>
|
||||||
<ng-component-outlet-complete-example></ng-component-outlet-complete-example>`
|
<ng-component-outlet-complete-example></ng-component-outlet-complete-example>
|
||||||
|
<hr/>
|
||||||
|
<ng-component-outlet-other-module-example></ng-component-outlet-other-module-example>`
|
||||||
})
|
})
|
||||||
class ExampleApp {
|
class ExampleApp {
|
||||||
}
|
}
|
||||||
|
@ -73,11 +96,19 @@ class ExampleApp {
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [BrowserModule],
|
imports: [BrowserModule],
|
||||||
declarations: [
|
declarations: [
|
||||||
ExampleApp, NgTemplateOutletSimpleExample, NgTemplateOutletCompleteExample, HelloWorld,
|
ExampleApp, NgTemplateOutletSimpleExample, NgTemplateOutletCompleteExample,
|
||||||
CompleteComponent
|
NgTemplateOutletOtherModuleExample, HelloWorld, CompleteComponent
|
||||||
],
|
],
|
||||||
entryComponents: [HelloWorld, CompleteComponent],
|
entryComponents: [HelloWorld, CompleteComponent],
|
||||||
bootstrap: [ExampleApp]
|
bootstrap: [ExampleApp]
|
||||||
})
|
})
|
||||||
export class AppModule {
|
export class AppModule {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
imports: [CommonModule],
|
||||||
|
declarations: [OtherModuleComponent],
|
||||||
|
entryComponents: [OtherModuleComponent]
|
||||||
|
})
|
||||||
|
export class OtherModule {
|
||||||
|
}
|
|
@ -118,13 +118,14 @@ export declare class NgClass implements DoCheck {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @experimental */
|
/** @experimental */
|
||||||
export declare class NgComponentOutlet implements OnChanges {
|
export declare class NgComponentOutlet implements OnChanges, OnDestroy {
|
||||||
componentRef: ComponentRef<any>;
|
|
||||||
ngComponentOutlet: Type<any>;
|
ngComponentOutlet: Type<any>;
|
||||||
ngComponentOutletContent: any[][];
|
ngComponentOutletContent: any[][];
|
||||||
ngComponentOutletInjector: Injector;
|
ngComponentOutletInjector: Injector;
|
||||||
constructor(_cmpFactoryResolver: ComponentFactoryResolver, _viewContainerRef: ViewContainerRef);
|
ngComponentOutletNgModuleFactory: NgModuleFactory<any>;
|
||||||
|
constructor(_viewContainerRef: ViewContainerRef);
|
||||||
ngOnChanges(changes: SimpleChanges): void;
|
ngOnChanges(changes: SimpleChanges): void;
|
||||||
|
ngOnDestroy(): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @stable */
|
/** @stable */
|
||||||
|
|
Loading…
Reference in New Issue