所以我想知道是否有办法传递 ng 模板并生成所有内容以包含插值中使用的变量?
此外,我仍然是角度的新手,所以除了删除html元素之外,我还需要担心删除其他任何东西吗?
在此结束时,将有一个指向 stackblitz.com 存储库的链接,该存储库将包含如下所示的所有代码。
以下是我实现指令的 src/app/app.component.html 代码:
<hello name="{{ name }}"></hello>
<p>
Start editing to see some magic happen :)
</p>
<!-- popup/popup.directive.ts contains the code i used in button tag -->
<button PopupDir="" body="this is a hardcoded message that is passed to popup box"> simple
</button>
<ng-template #Complicated="">
<div style="background-color: red;">
a little more complicated but simple and still doable
</div>
</ng-template>
<button PopupDir="" [body]="Complicated">
complicated
</button>
<ng-template #EvenMoreComplicated="">
<!-- name and data isn't being passed i need help here-->
<div style="background-color: green; min-height: 100px; min-width:100px;">
{{name}} {{data}}
</div>
</ng-template>
<button PopupDir="" [body]="EvenMoreComplicated">
more complicated
</button>
以下是我的src/app/popup/popup.directive.ts
import { Directive, Input, TemplateRef, ViewContainerRef, HostListener } from '@angular/core'
@Directive({
selector: 'PopupDir, [PopupDir]'
})
export class Popup {
@Input() body: string | TemplateRef<any>;
viewContainer: ViewContainerRef;
popupElement: HTMLElement;
//i dont know if i need this
constructor (viewContainer: ViewContainerRef) {
this.viewContainer = viewContainer;
}
//adds onlick rule to parent tag
@HostListener('click')
onclick () {
this.openPopup();
}
openPopup() {
//Pcreate pupup html programatically
this.popupElement = this.createPopup();
//insert it in the dom
const lastChild = document.body.lastElementChild;
lastChild.insertAdjacentElement('afterend', this.popupElement);
}
createPopup(): HTMLElement {
const popup = document.createElement('div');
popup.classList.add('popupbox');
//if you click anywhere on popup it will close/remove itself
popup.addEventListener('click', (e: Event) => this.removePopup());
//if statement to determine what type of "body" it is
if (typeof this.body === 'string')
{
popup.innerText = this.body;
} else if (typeof this.body === 'object')
{
const appendElementToPopup = (element: any) => popup.appendChild(element);
//this is where i get stuck on how to include the context and then display the context/data that is passed by interpolation in ng-template
this.body.createEmbeddedView(this.viewContainer._view.context).rootNodes.forEach(appendElementToPopup);
}
return popup;
}
removePopup() {
this.popupElement.remove();
}
}
这是显示我的问题的存储库的链接: https://stackblitz.com/edit/popupproblem
首先,让我们考虑如何将上下文传递给嵌入式视图。你写道:
this.body.createEmbeddedView(this.viewContainer._view.context)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
您的Popup
组件托管在视图中AppComponent
因此this.viewContainer._view.context
将AppComponent
实例。但我想让你说的是:
1( 嵌入式视图已经可以访问定义ng-template
的模板的范围。
2(如果我们传递上下文,那么它应该只通过模板引用变量使用。
this.body.createEmbeddedView(this.viewContainer._view.context)
||
/
this.body.createEmbeddedView({
name = 'Angular';
data = 'this should be passed too'
})
||
/
<ng-template #EvenMoreComplicated let-name="name" let-data="data">
{{name}} {{data}}
因此,在这种情况下,您不需要传递上下文,因为它已经存在。
this.body.createEmbeddedView({})
||
/
<ng-template #EvenMoreComplicated>
{{name}} {{data}}
为什么 UI 不更新?
角度变化检测机制依赖于视图树。
AppComponent_View
/
ChildComponent_View EmbeddedView
|
SubChildComponent_View
我们看到有两种视图:组件视图和嵌入式视图。TemplateRef
(ng-template(表示嵌入视图。
当 Angular 想要更新 UI 时,它只需通过该视图两个检查绑定。
现在让我们提醒一下如何通过低级 API 创建嵌入式视图:
TemplateRef.createEmbeddedView
ViewContainerRef.createEmbeddedView
它们之间的主要区别在于前者只是创建嵌入式视图,而后者创建嵌入式视图并将其添加到角度变化检测树中。这样,嵌入式视图成为更改检测树的一部分,我们可以看到更新的绑定。
是时候看看你的代码了:
this.body.createEmbeddedView(this.viewContainer._view.context).rootNodes.forEach(appendElementToPopup);
应该清楚的是,您使用的是第一种方法。这意味着您必须自己处理更改检测:手动调用viewRef.detectChanges()
或附加到树。
简单的解决方案可能是:
const view = this.body.createEmbeddedView({});
view.detectChanges();
view.rootNodes.forEach(appendElementToPopup);
堆栈闪电战示例
但它只会检测一次更改。我们可以在每个Popup.ngDoCheck()
钩子上调用detectChanges
方法,但 Angular 本身使用了一种更简单的方法。
const view = this.viewContainer.createEmbeddedView(this.body);
view.rootNodes.forEach(appendElementToPopup);
我们使用了创建嵌入式视图的第二种方法,以便 Angular 本身会自动检查模板。
我仍然对角度不熟悉,所以除了删除 html 元素之外,我还要 需要担心删除其他任何东西吗?
我认为我们还应该在关闭弹出窗口时销毁嵌入式视图。
removePopup() {
this.viewContainer.clear();
...
}
最终堆栈闪电战示例