通过单步调试的方式学习 Angular 中带有选择器的内容投影使用方式
data:image/s3,"s3://crabby-images/d5c2f/d5c2f1f66aeb398783c7d649e1af3519342d0644" alt="通过单步调试的方式学习 Angular 中带有选择器的内容投影使用方式"
问题描述
我创建了一个 selector 为 app-content-section
的 Component,用于容纳内容投影(content projection
):
这个 Component 的模板区域,分别定义了三个 div 区域,里面包含了对应的内容投影占位符 ng-content
:
data:image/s3,"s3://crabby-images/ecb6f/ecb6fd0cc6962859f87ee4abbcc291925ab96a6e" alt=""
第一个绿色区域接收所有 div 标签,第二个蓝色区域接收所有包含了 css class content-class
的标签,第三个区域接收所有 name 属性值为 test 的标签。
我们来看看消费这个 Component 的代码:
app-content-section
中包含两个 div 元素,故最后的绿色区域里,包含了两个 div 值:
data:image/s3,"s3://crabby-images/256a4/256a4afa101a97936279dfb1215b2879f16f1100" alt=""
蓝色区域内只显示了一个 p 标签,因为这是消费者传入的元素里,唯一一个带有 content-class
类的元素。
data:image/s3,"s3://crabby-images/3d0a0/3d0a0d71ec8b5b863250e30f9527993f82e0c695" alt=""
接下来的问题是,消费者传入的元素里,第一个 div 元素和最后一个 p 元素的 name
属性值都为 test
,为什么只有最后一个 p 标签,被投影到 app-content-section
里呢?
data:image/s3,"s3://crabby-images/d74b5/d74b5f6647c3f93c12c2ae98da27dac9568e6c41" alt=""
问题分析
第一个 div 标签同时满足第一条和第三条内容投影规则,因此其被投影到第一个 ng-content
之后,就不会再重复被投影了。
我们可以做一个测试,把提供内容投影场所的 Component 内的第一个和第三个 ng-content
调换一下顺序:
data:image/s3,"s3://crabby-images/cfbba/cfbbad52a8e1f3635552addcd3d8d354a7ee800a" alt=""
这次的测试结果,红色区域出现的两个元素,其 name 属性值都为 test. 而虽然绿色区域容纳的是被投射的 div 元素,但是因为 name = test 的 div 元素,已经被优先投射到红色区域,所以它不会再被重复投射了。
data:image/s3,"s3://crabby-images/fc488/fc4881ba839a90e78a28e74688ab451df38a3648" alt=""
总结
通过单步调试 Angular content projection 的相关代码,也能确认上述逻辑。
当 Component 的模板文件被解析,遇到 ng-content
时:
data:image/s3,"s3://crabby-images/40c31/40c318b803803f8e981513e3022dfa37f360daef" alt=""
触发 ɵɵprojection
函数。
data:image/s3,"s3://crabby-images/e789e/e789ec4c4e31c0a78f8b1f6ec6c9daf86caf5635" alt=""
applyProjectionRecursive
函数里的 nodeToProject
,就是需要被投影的 DOM 节点:
data:image/s3,"s3://crabby-images/8e097/8e097075137ca890fbe03bf315889444e68782ba" alt=""
可以看到这里只有包含了 name = test 属性值的 div 被 Angular 框架解析成需要被投影的节点。
版权声明: 本文为 InfoQ 作者【Jerry Wang】的原创文章。
原文链接:【http://xie.infoq.cn/article/374751437142c894dccba6fc6】。文章转载请联系作者。
评论