Versão angular que usei - Angular 4.2.0
O Angular 4 veio com o ComponentFactoryResolver para carregar componentes no tempo de execução. Este é um tipo da mesma implementação do $ compile no Angular 1.0 que atende à sua necessidade
Neste exemplo abaixo, estou carregando o componente ImageWidget dinamicamente em um DashboardTileComponent
Para carregar um componente, você precisa de uma diretiva que possa ser aplicada ao ng-template que ajudará a colocar o componente dinâmico
WidgetHostDirective
import { Directive, ViewContainerRef } from '@angular/core';
@Directive({
selector: '[widget-host]',
})
export class DashboardTileWidgetHostDirective {
constructor(public viewContainerRef: ViewContainerRef) {
}
}
essa diretiva injeta ViewContainerRef para obter acesso ao contêiner de exibição do elemento que hospedará o componente adicionado dinamicamente.
DashboardTileComponent (componente de suporte de local para renderizar o componente dinâmico)
Este componente aceita uma entrada proveniente dos componentes pai ou você pode carregar do seu serviço com base na sua implementação. Este componente está desempenhando o papel principal de resolver os componentes em tempo de execução. Nesse método, você também pode ver um método chamado renderComponent () que, finalmente, carrega o nome do componente de um serviço e é resolvido com ComponentFactoryResolver e, finalmente, configurando dados para o componente dinâmico.
import { Component, Input, OnInit, AfterViewInit, ViewChild, ComponentFactoryResolver, OnDestroy } from '@angular/core';
import { DashboardTileWidgetHostDirective } from './DashbardWidgetHost.Directive';
import { TileModel } from './Tile.Model';
import { WidgetComponentService } from "./WidgetComponent.Service";
@Component({
selector: 'dashboard-tile',
templateUrl: 'app/tile/DashboardTile.Template.html'
})
export class DashboardTileComponent implements OnInit {
@Input() tile: any;
@ViewChild(DashboardTileWidgetHostDirective) widgetHost: DashboardTileWidgetHostDirective;
constructor(private _componentFactoryResolver: ComponentFactoryResolver,private widgetComponentService:WidgetComponentService) {
}
ngOnInit() {
}
ngAfterViewInit() {
this.renderComponents();
}
renderComponents() {
let component=this.widgetComponentService.getComponent(this.tile.componentName);
let componentFactory = this._componentFactoryResolver.resolveComponentFactory(component);
let viewContainerRef = this.widgetHost.viewContainerRef;
let componentRef = viewContainerRef.createComponent(componentFactory);
(<TileModel>componentRef.instance).data = this.tile;
}
}
DashboardTileComponent.html
<div class="col-md-2 col-lg-2 col-sm-2 col-default-margin col-default">
<ng-template widget-host></ng-template>
</div>
WidgetComponentService
Esta é uma fábrica de serviços para registrar todos os componentes que você deseja resolver dinamicamente
import { Injectable } from '@angular/core';
import { ImageTextWidgetComponent } from "../templates/ImageTextWidget.Component";
@Injectable()
export class WidgetComponentService {
getComponent(componentName:string) {
if(componentName==="ImageTextWidgetComponent"){
return ImageTextWidgetComponent
}
}
}
ImageTextWidgetComponent (componente que estamos carregando em tempo de execução)
import { Component, OnInit, Input } from '@angular/core';
@Component({
selector: 'dashboard-imagetextwidget',
templateUrl: 'app/templates/ImageTextWidget.html'
})
export class ImageTextWidgetComponent implements OnInit {
@Input() data: any;
constructor() { }
ngOnInit() { }
}
Adicionar Finalmente, adicione este ImageTextWidgetComponent ao seu módulo de aplicativo como entryComponent
@NgModule({
imports: [BrowserModule],
providers: [WidgetComponentService],
declarations: [
MainApplicationComponent,
DashboardHostComponent,
DashboardGroupComponent,
DashboardTileComponent,
DashboardTileWidgetHostDirective,
ImageTextWidgetComponent
],
exports: [],
entryComponents: [ImageTextWidgetComponent],
bootstrap: [MainApplicationComponent]
})
export class DashboardModule {
constructor() {
}
}
TileModel
export interface TileModel {
data: any;
}
Referência original do meu blog
Documentação oficial
Baixar código-fonte de amostra