博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Angular 2 TemplateRef & ViewContainerRef
阅读量:6334 次
发布时间:2019-06-22

本文共 7275 字,大约阅读时间需要 24 分钟。

阅读 Angular 6/RxJS 最新教程,请访问

TemplateRef

在介绍 TemplateRef 前,我们先来了解一下 HTML - <template> 。模板元素是一种机制,允许包含加载页面时不渲染,但又可以随后通过 JavaScript 进行实例化的客户端内容。我们可以将模板视作为存储在页面上稍后使用的一小段内容。

在 HTML5 标准引入 template 模板元素之前,我们都是使用 <script> 标签进行客户端模板的定义,具体如下:

对于支持 HTML5 template 模板元素的浏览器,我们可以这样创建客户端模板:

下面我们来看一下 HTML5 template 模板元素的使用示例:

HTML5 Template Element Demo

HTML5 Template Element Demo

以上代码运行后,在浏览器中我们会看到以下内容:

HTML5 Template Element DemoI am span in template

而当我们注释掉 tplContainer.appendChild(tplNode) 语句时,刷新浏览器后看到的是:

HTML5 Template Element Demo

这说明页面中 <template> 模板元素中的内容,如果没有进行处理对用户来说是不可见的。Angular 2 中,<template> 模板元素主要应用在结构指令中,此外在 文章中我们也介绍了 <template> 模板元素和自定义结构指令,接下来我们先来介绍一下本文中的第一个主角 - TemplateRef:

import {Component, TemplateRef, ViewChild, AfterViewInit} from '@angular/core';@Component({  selector: 'my-app',  template: `    

Welcome to Angular World

`,})export class AppComponent { name: string = 'Semlinker'; @ViewChild('tpl') tpl: TemplateRef
; ngAfterViewInit() { console.dir(this.tpl); }}

上述代码运行后的控制台的输出结果如下:

图片描述

从上图中,我们发现 @Component template 中定义的 <template> 模板元素,渲染后被替换成 comment 元素,其内容为 "template bindings={}" 。此外我们通过 @ViewChild 获取的模板元素,是 TemplateRef_ 类的实例,接下来我们来研究一下 TemplateRef_ 类:

TemplateRef_

// @angular/core/src/linker/template_ref.d.tsexport declare class TemplateRef_
extends TemplateRef
{ private _parentView; private _nodeIndex; private _nativeElement; constructor(_parentView: AppView
, _nodeIndex: number, _nativeElement: any); createEmbeddedView(context: C): EmbeddedViewRef
; elementRef: ElementRef;}

TemplateRef

// @angular/core/src/linker/template_ref.d.ts// 用于表示内嵌的template模板,能够用于创建内嵌视图(Embedded Views)export declare abstract class TemplateRef
{ elementRef: ElementRef; abstract createEmbeddedView(context: C): EmbeddedViewRef
;}

(备注:抽象类与普通类的区别是抽象类有包含抽象方法,不能直接实例化抽象类,只能实例化该抽象类的子类)

我们已经知道 <template> 模板元素,渲染后被替换成 comment 元素,那么应该如何显示我们模板中定义的内容呢 ?我们注意到了 TemplateRef 抽象类中定义的 createEmbeddedView

抽象方法,该方法的返回值是 EmbeddedViewRef 对象。那好我们马上来试一下:

import {Component, TemplateRef, ViewChild, AfterViewInit} from '@angular/core';@Component({  selector: 'my-app',  template: `    

Welcome to Angular World

`,})export class AppComponent { name: string = 'Semlinker'; @ViewChild('tpl') tpl: TemplateRef
; ngAfterViewInit() { let embeddedView = this.tpl.createEmbeddedView(null); console.dir(embeddedView); }}

上述代码运行后的控制台的输出结果如下:

图片描述

从图中我们可以知道,当调用 createEmbeddedView 方法后返回了 ViewRef_ 视图对象。该视图对象的 rootNodes 属性包含了 <template> 模板中的内容。在上面的例子中,我们知道了 TemplateRef 实例对象中的 elementRef 属性封装了我们的 comment 元素,那么我们可以通过 insertBefore 方法来创建我们模板中定义的内容。

import { Component, TemplateRef, ViewChild, AfterViewInit } from '@angular/core';@Component({  selector: 'my-app',  template: `    

Welcome to Angular World

`,})export class AppComponent { name: string = 'Semlinker'; @ViewChild('tpl') tpl: TemplateRef
; ngAfterViewInit() { // 页面中的
元素 let commentElement = this.tpl.elementRef.nativeElement; // 创建内嵌视图 let embeddedView = this.tpl.createEmbeddedView(null); // 动态添加子节点 embeddedView.rootNodes.forEach((node) => { commentElement.parentNode .insertBefore(node, commentElement.nextSibling); }); }}

成功运行上面的代码后,在浏览器中我们会看到以下内容:

Welcome to Angular WorldI am span in template

现在我们来回顾一下,上面的处理步骤:

  • 创建内嵌视图(embedded view)
  • 遍历内嵌视图中的 rootNodes,动态的插入 node

虽然我们已经成功的显示出 template 模板元素中的内容,但发现整个流程还是太复杂了,那有没有简单地方式呢 ?是时候介绍本文中第二个主角 - ViewContainerRef。

ViewContainerRef

我们先来检验一下它的能力,然后再来好好地分析它。具体示例如下:

import { Component, TemplateRef, ViewChild, ViewContainerRef, AfterViewInit } from '@angular/core';@Component({  selector: 'my-app',  template: `    

Welcome to Angular World

`,})export class AppComponent { name: string = 'Semlinker'; @ViewChild('tpl') tplRef: TemplateRef
; @ViewChild('tpl', { read: ViewContainerRef }) tplVcRef: ViewContainerRef; ngAfterViewInit() { // console.dir(this.tplVcRef); (1) this.tplVcRef.createEmbeddedView(this.tplRef); }}

移除上面代码中的注释,即可在控制台看到以下的输出信息:

图片描述

而在浏览器中我们会看到以下内容:

Welcome to Angular WorldI am span in template

接下来我们来看一下 ViewContainerRef_ 类:

// @angular/core/src/linker/view_container_ref.d.ts// 用于表示一个视图容器,可添加一个或多个视图export declare class ViewContainerRef_ implements ViewContainerRef {    ...    length: number; // 返回视图容器中已存在的视图个数    element: ElementRef;    injector: Injector;    parentInjector: Injector;      // 基于TemplateRef创建内嵌视图,并自动添加到视图容器中,可通过index设置    // 视图添加的位置    createEmbeddedView
(templateRef: TemplateRef
, context?: C, index?: number): EmbeddedViewRef
; // 基 ComponentFactory创建组件视图 createComponent
(componentFactory: ComponentFactory
, index?: number, injector?: Injector, projectableNodes?: any[][]): ComponentRef
; insert(viewRef: ViewRef, index?: number): ViewRef; move(viewRef: ViewRef, currentIndex: number): ViewRef; indexOf(viewRef: ViewRef): number; remove(index?: number): void; detach(index?: number): ViewRef; clear(): void;}

通过源码我们可以知道通过 ViewContainerRef_ 实例,我们可以方便地操作视图,也可以方便地基于 TemplateRef 创建视图。现在我们来总结一下 TemplateRef 与 ViewContainerRef。

TemplateRef:用于表示内嵌的 template 模板元素,通过 TemplateRef 实例,我们可以方便创建内嵌视图(Embedded Views),且可以轻松地访问到通过 ElementRef 封装后的 nativeElement。需要注意的是组件视图中的 template 模板元素,经过渲染后会被替换成 comment 元素。

ViewContainerRef:用于表示一个视图容器,可添加一个或多个视图。通过 ViewContainer

Ref 实例,我们可以基于 TemplateRef 实例创建内嵌视图,并能指定内嵌视图的插入位置,也可以方便对视图容器中已有的视图进行管理。简而言之,ViewContainerRef 的主要作用是创建和管理内嵌视图或组件视图。

我有话说

1.Angular 2 支持的 View(视图) 类型有哪几种 ?

  • Embedded Views - Template 模板元素
  • Host Views - Component 组件

1.1 如何创建 Embedded View

ngAfterViewInit() {    let view = this.tpl.createEmbeddedView(null);}

1.2 如何创建 Host View

constructor(private injector: Injector,    private r: ComponentFactoryResolver) {    let factory = this.r.resolveComponentFactory(AppComponent);    let componentRef = factory.create(injector);    let view = componentRef.hostView;}

2.Angular 2 Component 组件中定义的 <template> 模板元素为什么渲染后会被移除 ?

因为 <template> 模板元素,已经被 Angular 2 解析并封装成 TemplateRef 实例,通过 TemplateRef 实例,我们可以方便地创建内嵌视图(Embedded View),我们不需要像开篇中的例子那样,手动操作 <template> 模板元素。

3.ViewRef 与 EmbeddedViewRef 之间有什么关系 ?

ViewRef 用于表示 Angular View(视图),视图是可视化的 UI 界面。EmbeddedViewRef 继承于 ViewRef,用于表示 <template> 模板元素中定义的 UI 元素。

ViewRef

// @angular/core/src/linker/view_ref.d.tsexport declare abstract class ViewRef {    destroyed: boolean;    abstract onDestroy(callback: Function): any;}

EmbeddedViewRef

// @angular/core/src/linker/view_ref.d.tsexport declare abstract class EmbeddedViewRef
extends ViewRef { context: C; rootNodes: any[]; // 保存

总结

Angular 2 中 TemplateRef 与 ViewContainerRef 的概念对于初学者来说会比较羞涩难懂,本文从基本的 HTML 5 <template> 模板元素开始,介绍了如何操作和应用页面中定义的模板。然后通过实例介绍了 Angular 2 中 TemplateRef 和 ViewContainerRef 的定义和作用。希望通过这篇文章,读者能更好的理解 TemplateRef 与 ViewContainerRef。

转载地址:http://tgioa.baihongyu.com/

你可能感兴趣的文章
Web前端面试题目及答案汇总
查看>>
【windows】如何获取文件夹目录
查看>>
python3 time模块的引用
查看>>
java代码---------打印正三角形
查看>>
git(5) windows下 pycharm + git(github) ,在本地方便管理
查看>>
泛型集合
查看>>
react + redux 完整的项目,同时写一下个人感悟
查看>>
vue开发移动端总结
查看>>
JMeter添加压力机、下载文件
查看>>
c# 字符串加密解密
查看>>
20155301 信息安全系统设计基础第五次实验
查看>>
09-Python字典的基础
查看>>
linux 下简单的ftp客户端程序
查看>>
ASP.NET MVC3 Model验证总结
查看>>
自定义全局样式
查看>>
08 常见事件响应的实现方式对比
查看>>
php 验证邮箱的方法
查看>>
RCNN,Fast RCNN,Faster RCNN 的前生今世:(2)R-CNN
查看>>
Unity3D——相机跟随物体移动
查看>>
C#语言课程11月6日
查看>>