I'm developping a little UI components framework for my personal needs and for fun. I'm developping a Tab component and for testing purpose, I need to inject dynamically a component (TabContainerComponent) inside another component (TabComponent). Below, the code of my two components:
tab.component.ts:
import {Component, ContentChildren} from "@angular/core";
import {TabContainerComponent} from "./tabContainer.component";
@Component({
selector: 'tab',
templateUrl: 'tab.component.html'
})
export class TabComponent {
@ContentChildren(TabContainerComponent)
tabs: TabContainerComponent[];
}
tab.component.html:
<ul>
<li *ngFor="let tab of tabs">{{ tab.title }}</li>
</ul>
<div>
<div *ngFor="let tab of tabs">
<ng-container *ngTemplateOutlet="tab.template"></ng-container>
</div>
<ng-content></ng-content>
</div>
tabContainer.component.ts:
import {Component, Input} from "@angular/core";
@Component({
selector: 'tab-container',
template: '<ng-container></ng-container>'
})
export class TabContainerComponent {
@Input()
title: string;
@Input()
template;
}
I used a ComponentFactoryResolver and a ComponentFactory to create dynamically my new component (TabContainerComponent), and inject it in a placeholder inside my other component (TabContainer), in the addTab method:
app.component.ts:
import {
Component, ViewChild, ComponentFactoryResolver, ComponentFactory,
ComponentRef, TemplateRef, ViewContainerRef
} from '@angular/core';
import {TabContainerComponent} from "./tabContainer.component";
import {TabComponent} from "./tab.component";
@Component({
selector: 'app-root',
templateUrl: './app.component.html',
styleUrls: ['./app.component.css']
})
export class AppComponent {
title = 'app';
@ViewChild(TabComponent)
tab: TabComponent;
@ViewChild('tabsPlaceholder', {read: ViewContainerRef})
public tabsPlaceholder: ViewContainerRef;
@ViewChild('newTab')
newTab: TemplateRef<any>;
constructor(private resolver: ComponentFactoryResolver) {
}
addTab(): void {
let factory: ComponentFactory<TabContainerComponent> = this.resolver.resolveComponentFactory(TabContainerComponent);
let tab: ComponentRef<TabContainerComponent> = this.tabsPlaceholder.createComponent(factory);
tab.instance.title = "New tab";
tab.instance.template = this.newTab;
console.log('addTab() triggered');
}
}
The addMethod is triggered by clicking on the "Add tab" button:
app.component.html:
<button (click)="addTab()">Add tab</button>
<tab>
<tab-container title="Tab 1" [template]="tab1"></tab-container>
<tab-container title="Tab 2" [template]="tab2"></tab-container>
<ng-container #tabsPlaceholder></ng-container>
</tab>
<ng-template #tab1>T1 template</ng-template>
<ng-template #tab2>T2 template</ng-template>
<ng-template #newTab>
This is a new tab
</ng-template>
app.module.ts:
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { AppComponent } from './app.component';
import {TabContainerComponent} from "./tabContainer.component";
import {TabComponent} from "./tab.component";
@NgModule({
declarations: [
AppComponent,
TabComponent,
TabContainerComponent
],
imports: [
BrowserModule
],
providers: [],
bootstrap: [AppComponent],
entryComponents: [
TabContainerComponent
]
})
export class AppModule { }
When I click on the "Add tab" button, I'm able to see the console.log message and I'm able to see a new <tab-container> tag (but without any attribute, which is strange) inside the <tab> tag but Angular doesn't update the Tab component view (there is no <li> and <div> created). I tried also to check changes by implementing OnChanges interface in TabComponent class but without success.
Is anyone have an idea to solve my problem ?
P.S.: I don't want to use an array of TabContainer components in order to test the createComponent method.
Update:
Demo:
https://stackblitz.com/edit/angular-imeh71?embed=1&file=src/app/app.component.ts
答案 0 :(得分:2)
这是我可以使其工作的方式-
Tab.component.ts
将“标签”从TabContainerComponent数组更改为QueryList。
import { Component, ContentChildren, QueryList } from '@angular/core';
import { TabContainerComponent } from '../tab-container/tab-container.component';
@Component({
selector: 'app-tab',
templateUrl: 'tab.component.html'
})
export class TabComponent {
@ContentChildren(TabContainerComponent)
tabs: QueryList<TabContainerComponent>;
constructor() {}
}
在 app.component.html
中添加了新模板<ng-template #tabContainerTemplate>
<app-tab-container title="New Tab" [template]="newTab"></app-tab-container>
</ng-template>
app.component.ts
import {
Component,
ViewChild,
TemplateRef,
ViewContainerRef,
AfterViewInit,
ViewChildren,
QueryList,
ChangeDetectorRef
} from '@angular/core';
import { TabContainerComponent } from './tab-container/tab-container.component';
import { TabComponent } from './tab/tab.component';
@Component({
selector: 'app-root',
templateUrl: './app.component.html',
styleUrls: ['./app.component.css']
})
export class AppComponent implements AfterViewInit {
title = 'app';
changeId: string;
@ViewChild(TabComponent) tab: TabComponent;
@ViewChild('tabsPlaceholder', { read: ViewContainerRef })
public tabsPlaceholder: ViewContainerRef;
@ViewChild('tabContainerTemplate', { read: TemplateRef })
tabContainerTemplate: TemplateRef<null>;
@ViewChildren(TabContainerComponent)
tabList: QueryList<TabContainerComponent>;
constructor(private changeDetector: ChangeDetectorRef) {}
ngAfterViewInit() {}
addTab(): void {
this.tabsPlaceholder.createEmbeddedView(this.tabContainerTemplate);
this.tab.tabs = this.tabList;
this.changeDetector.detectChanges();
console.log('addTab() triggered');
}
}
为TabContainerComponent添加了ViewChildren查询。在addTab()中,使用createEmbeddedView添加了新的标签容器组件。
我认为TabComponent中的“ ContentChildren”查询应该由新添加的组件来更新,但事实并非如此。我已经尝试为TabCompoent中的查询列表订阅“更改”,但不会被触发。
但是我发现,每次添加新组件时,AppComponent中的“ ViewChildren”查询都会更新。因此,我已将应用程序组件的更新的QueryList分配给TabComponent的QueryList。
可用的演示程序here
答案 1 :(得分:0)
我已经修改了您的代码,检查它是否按预期显示在视图中。
对于动态组件创建,您需要使用templateRef创建嵌入式视图。
View容器提供用于创建,操作和删除动态信息的API 视图。
有关动态组件操作的更多信息,请检查以下内容:https://blog.angularindepth.com/working-with-dom-in-angular-unexpected-consequences-and-optimization-techniques-682ac09f6866
import {
Component, ViewChild, ComponentFactoryResolver, ComponentFactory,
ComponentRef, TemplateRef, ViewContainerRef,AfterViewInit
} from '@angular/core';
import {TabContainerComponent} from "./hello.component";
import {TapComponent} from "./tap/tap.component";
@Component({
selector: 'app-root',
templateUrl: './app.component.html',
styleUrls: ['./app.component.css']
})
export class AppComponent implements AfterViewInit {
title = 'app';
@ViewChild('vc',{read:ViewContainerRef}) vc:ViewContainerRef;
@ViewChild(TapComponent)
tab: TapComponent;
@ViewChild('tabsPlaceholder', {read: ViewContainerRef})
public tabsPlaceholder: ViewContainerRef;
@ViewChild('newTab')
newTab: TemplateRef<any>;
constructor(private resolver: ComponentFactoryResolver) {
}
ngAfterViewInit(){
}
addTab(): void {
let factory: ComponentFactory<TabContainerComponent> = this.resolver.resolveComponentFactory(TabContainerComponent);
let tab: ComponentRef<TabContainerComponent> = this.tabsPlaceholder.createComponent(factory);
tab.instance.title = "New tab";
tab.instance.template = this.newTab;
this.vc.createEmbeddedView(this.newTab);
console.log('addTab() triggered');
}
}