Кутова версія, яку я використав - Кутова 4.2.0
Кутовий 4 придумав ComponentFactoryResolver для завантаження компонентів під час виконання. Це свого роду реалізація $ compile в Angular 1.0, яка відповідає вашим потребам
У цьому нижче прикладі я динамічно завантажую компонент ImageWidget у DashboardTileComponent
Для завантаження компонента вам потрібна директива, яку ви можете застосувати до ng-template, який допоможе розмістити динамічний компонент
WidgetHostDirective
import { Directive, ViewContainerRef } from '@angular/core';
@Directive({
selector: '[widget-host]',
})
export class DashboardTileWidgetHostDirective {
constructor(public viewContainerRef: ViewContainerRef) {
}
}
ця директива вводить ViewContainerRef для отримання доступу до контейнера перегляду елемента, в якому буде розміщений динамічно доданий компонент.
DashboardTileComponent (компонент тримача для розміщення динамічного компонента)
Цей компонент приймає вхід, який надходить від батьківських компонентів, або ви можете завантажити з вашої служби на основі вашої реалізації. Цей компонент виконує головну роль для вирішення компонентів під час виконання. У цьому методі ви також можете побачити метод з назвою renderComponent (), який в кінцевому підсумку завантажує ім'я компонента з сервісу і вирішує з ComponentFactoryResolver і, нарешті, встановлює дані на динамічний компонент.
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
Це фабрика послуг для реєстрації всіх компонентів, які ви хочете динамічно вирішити
import { Injectable } from '@angular/core';
import { ImageTextWidgetComponent } from "../templates/ImageTextWidget.Component";
@Injectable()
export class WidgetComponentService {
getComponent(componentName:string) {
if(componentName==="ImageTextWidgetComponent"){
return ImageTextWidgetComponent
}
}
}
ImageTextWidgetComponent (компонент, який ми завантажуємо під час виконання)
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() { }
}
Додати Нарешті, додайте цей ImageTextWidgetComponent у свій модуль додатка як 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;
}
Оригінальна довідка з мого блогу
Офіційна документація
Завантажте зразковий вихідний код