Angular 内容投影 content projection
问题描述
我使用如下代码测试一个最简单的 Angular 内容投影场景:
import { Component } from '@angular/core';
@Component({
selector: 'app-zippy-basic',
template: `
<h2>Single-slot content projection</h2>
<ng-container></ng-container>
`
})
export class ZippyBasicComponent {}
data:image/s3,"s3://crabby-images/3f1fd/3f1fd0afafefc4d1f756f5706d75061bd153c10d" alt=""
我期望 app-zippy-basic
的使用者,将传入的 content
,投射到元素 ng-container
内部。
消费代码如下图所示:
<app-zippy-basic>
<p>Is content projection cool?</p>
<div>ok</div>
</app-zippy-basic>
然而运行时,我在渲染出的页面里,根本看不到 Is content projection cool?
的显示。
data:image/s3,"s3://crabby-images/3b371/3b3717c7bd313895921cf4198d2c5e3bcf9586ea" alt=""
问题分析
打开 Chrome 开发者工具,发现 app-zippy-basic
内部只有一个 comment 节点:ng-container
data:image/s3,"s3://crabby-images/03465/034657f7eeab8cea800aba4c12ffca4bfdf6735f" alt=""
我们在提供内容投影插槽的 Component 的 ng-container 之间随便键入一些字符串,例如 DIV:
data:image/s3,"s3://crabby-images/fae78/fae784e37558acd6f50f761466fe48fe80bbdef1" alt=""
渲染后发现,ng-container 没能按照我们期望的工作。
data:image/s3,"s3://crabby-images/4cf54/4cf54b8f6f3121b637ab1a0e1e57d930fc71f532" alt=""
通过查阅 Angular 官网,发现这里把 ng-container 和 ng-content 弄混淆了。此处应该使用 ng-content.
解决方案
ng-content
之间不允许再插入其他元素。仅仅充当一个占位符的角色。
data:image/s3,"s3://crabby-images/f299c/f299ca36558596d45f62eae34109c391840e4f3f" alt=""
使用 ng-content
之后问题消失。这里我们可以通过单步调试的方式,搞清楚被投影的内容是如何插入到 ng-content 占位符里的。
data:image/s3,"s3://crabby-images/3257e/3257e96ddea57122dfa5b95913de7c1212f28e0a" alt=""
选择 app-zippy-basic
,给其设置一个 dom 断点,类型为 subtree modification
:
data:image/s3,"s3://crabby-images/0ef3d/0ef3da49addc26352bc799ca8242b8750f485557" alt=""
首先执行 template 函数,将 Component 本身的 h2 元素,进行 HTML 源代码的渲染:
data:image/s3,"s3://crabby-images/5c58e/5c58e97f13c38ed01d428ef5b980b01c37455dd0" alt=""
data:image/s3,"s3://crabby-images/9ceb9/9ceb9665e52270466e8049e83ab537818772b0aa" alt=""
接下来是 Angular 处理整个内容投影逻辑的关键:
data:image/s3,"s3://crabby-images/7c2bb/7c2bb4e61ce509269e9bfd00e4e7997c80a1057c" alt=""
遇到 ng-content
,则调用函数 ɵɵprojection
:
data:image/s3,"s3://crabby-images/211f4/211f4aba3963fd02165d4b2e294b34237d4eee64" alt=""
转而调用 applyProjection
:
data:image/s3,"s3://crabby-images/168b8/168b8626f70d70b455f8f45afd99cad1074a509f" alt=""
parent node 即是指定义了 ng-content
的 Component 对应的 dom 元素:
data:image/s3,"s3://crabby-images/38ed7/38ed7381f06982d717269084bdebb4c6cfe4a9f5" alt=""
rNode 即是被插入到 app-zippy-basic
中的节点,即 app-zippy-basic
的消费者。
data:image/s3,"s3://crabby-images/6ec6c/6ec6c0ae5f5f6c7a4313a83b83f05c012ceede2d" alt=""
消费语法:
<app-zippy-basic>
此处插入你想投影到 app-zippy-basic 内部的 HTML 源代码
</app-zippy-basic>