我通过选择预先存在的组件创建了动态组件实例。例如,
@Component({
selector: 'dynamic-component',
template: `<div #container><ng-content></ng-content></div>`
})
export class DynamicComponent {
@ViewChild('container', {read: ViewContainerRef}) container: ViewContainerRef;
public addComponent(ngItem: Type<WidgetComponent>,selectedPlugin:Plugin): WidgetComponent {
let factory = this.compFactoryResolver.resolveComponentFactory(ngItem);
const ref = this.container.createComponent(factory);
const newItem: WidgetComponent = ref.instance;
newItem.pluginId = Math.random() + '';
newItem.plugin = selectedPlugin;
this._elements.push(newItem);
return newItem;
}
}
我之前存在的组件是ChartWidget和PatientWidget,它扩展了我想要在容器中添加的类WidgetComponent。例如,
@Component({
selector: 'chart-widget',
templateUrl: 'chart-widget.component.html',
providers: [{provide: WidgetComponent, useExisting: forwardRef(() => ChartWidget) }]
})
export class ChartWidget extends WidgetComponent implements OnInit {
constructor(ngEl: ElementRef, renderer: Renderer) {
super(ngEl, renderer);
}
ngOnInit() {}
close(){
console.log('close');
}
refresh(){
console.log('refresh');
}
...
}
chart-widget.compoment.html(使用primeng Panel)
<p-panel [style]="{'margin-bottom':'20px'}">
<p-header>
<div class="ui-helper-clearfix">
<span class="ui-panel-title" style="font-size:14px;display:inline-block;margin-top:2px">Chart Widget</span>
<div class="ui-toolbar-group-right">
<button pButton type="button" icon="fa-window-minimize" (click)="minimize()"</button>
<button pButton type="button" icon="fa-refresh" (click)="refresh()"></button>
<button pButton type="button" icon="fa-expand" (click)="expand()" ></button>
<button pButton type="button" (click)="close()" icon="fa-window-close"></button>
</div>
</div>
</p-header>
some data
</p-panel>
data-widget.compoment.html(与使用primeng Panel的chart-widget相同)
@Component({
selector: 'data-widget',
templateUrl: 'data-widget.component.html',
providers: [{provide: WidgetComponent, useExisting: forwardRef(() =>DataWidget) }]
})
export class DataWidget extends WidgetComponent implements OnInit {
constructor(ngEl: ElementRef, renderer: Renderer) {
super(ngEl, renderer);
}
ngOnInit() {}
close(){
console.log('close');
}
refresh(){
console.log('refresh');
}
...
}
WidgetComponent.ts
@Component({
selector: 'widget',
template: '<ng-content></ng-content>'
})
export class WidgetComponent{
}
现在,我通过以下方式从现有组件中选择一个组件(例如chart-widget和data-widget)来添加组件,并将实例存储到一个数组中。
@Component({
templateUrl: 'main.component.html',
entryComponents: [ChartWidget, DataWidget],
})
export class MainComponent implements OnInit {
private elements: Array<WidgetComponent>=[];
private WidgetClasses = {
'ChartWidget': ChartWidget,
'DataWidget': DataWidget
}
@ViewChild(DynamicComponent) dynamicComponent: DynamicComponent;
addComponent(): void{
let ref= this.dynamicComponent.addComponent(this.WidgetClasses[this.selectedComponent], this.selectedComponent);
this.elements.push(ref);
this.dynamicComponent.resetContainer();
}
}
现在,我在main.component.html中使用innerHtml渲染组件时遇到问题。它渲染html但我无法使用按钮点击事件或其他事件。我也尝试使用primeng渲染图表,但它也不起作用。
main.component.html
<dynamic-component [hidden]="true" ></dynamic-component>
<widget *ngFor="let item of elements">
<div [innerHTML]="item._ngEl.nativeElement.innerHTML | sanitizeHtml">
</div>
</widget>
我还实现了一个sanitizeHtml管道,但它给出了相同的结果。所以,据我所知,innerHTML只显示html数据,但我不能使用任何按钮事件以及js图表。我还尝试在标记下显示此{{item}}项。但它显示为文本[对象对象]。那么,任何人都可以为它提供解决方案吗?如何渲染允许按钮事件和js图表的组件?感谢。
编辑:在此处查看我的Plunker https://plnkr.co/edit/lugU2pPsSBd3XhPHiUP1?p=preview 你可以在这里看到,可以动态添加图表或数据小部件,我正在使用innerHTML显示它。所以,按钮事件在这里不起作用。如果我编码像{{item}}那么它会显示[object object]文本。您还可以在控制台中看到组件数组数据。主要问题是,如何激活其上的按钮事件(例如,如果我点击关闭或刷新按钮,那么它将调用相关功能)?
答案 0 :(得分:1)
我会创建结构指令,如:
<强> view.directive.ts 强>
WHERE
然后
<强> app.component.ts 强>
.
和
<强> app.component.html 强>
IS NULL LIMIT 5 OFFSET 5;
Empty set (0.00 sec)
所以我刚刚将动态组件容器的视图移动到了所需的位置。
<强> Plunker Example 强>