中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

angular之ng-template模板加載

發布時間:2020-09-13 08:47:46 來源:腳本之家 閱讀:434 作者:OnePiece索隆 欄目:web開發

本文介紹了angular之ng-template模板加載,分享給大家,具體如下:

html5中的template

template標簽的含義:HTML <template>元素是一種用于保存客戶端內容的機制,該內容在頁面加載時是不可見的,但可以在運行時使用JavaScript進行實例化,可以將一個模板視為正在被存儲以供隨后在文檔中使用的一個內容片段。

angular之ng-template模板加載

屬性

此元素僅包含全局屬性和只讀的 content 屬性,通過content 可以讀取模板內容,而且可以通過判斷 content 屬性是否存在來判斷瀏覽器是否支持 <template> 元素。

示例

html

<table id="producttable">
 <thead>
  <tr>
   <td>UPC_Code</td>
   <td>Product_Name</td>
  </tr>
 </thead>
 <tbody>
  <!-- 現有數據可以可選地包括在這里 -->
 </tbody>
</table>

<template id="productrow">
 <tr>
  <td class="record"></td>
  <td></td>
 </tr>
</template>

js

// 通過檢查來測試瀏覽器是否支持HTML模板元素 
// 用于保存模板元素的內容屬性。
if ('content' in document.createElement('template')) {

 // 使用現有的HTML tbody實例化表和該行與模板
 let t = document.querySelector('#productrow'),
 td = t.content.querySelectorAll("td");
 td[0].textContent = "1235646565";
 td[1].textContent = "Stuff";

 // 克隆新行并將其插入表中
 let tb = document.getElementsByTagName("tbody");
 let clone = document.importNode(t.content, true);
 tb[0].appendChild(clone);
 
 // 創建一個新行
 td[0].textContent = "0384928528";
 td[1].textContent = "Acme Kidney Beans";

 // 克隆新行并將其插入表中
 let clone2 = document.importNode(t.content, true);
 tb[0].appendChild(clone2);

} else {
 // 找到另一種方法來添加行到表,因為不支持HTML模板元素。
}

代碼運行后,結果將是一個包含(由 JavaScript 生成)兩個新行的 HTML 表格:

UPC_Code  Product_Name
1235646565 Stuff
0384928528 Acme Kidney Beans

注釋掉 tb[0].appendChild(clone);和tb[0].appendChild(clone2);,運行代碼,只會看到:
UPC_Code Product_Name

說明template元素中的內容如果不經過處理,瀏覽器是不會渲染的。

angular中的ng-template

<ng-template>是一個 Angular 元素,它永遠不會直接顯示出來。在渲染視圖之前,Angular 會把<ng-template>及其內容替換為一個注釋。

以ngIf為例:

angular之ng-template模板加載

<ng-template> 模板元素與html5的template元素一樣,需要被特殊處理后才能渲染。ng主要是通過類TemplateRef和ViewContainerRef實現的。

通過閱讀ngIf源碼學習如何運用<ng-template>

在使用ngIf 指令時我們并未發現ng-template的身影,這是因為"*"(星號)語法糖的原因,這個簡寫方法是一個微語法,而不是通常的模板表達式, Angular會解開這個語法糖,變成一個<ng-template>標記,包裹著宿主元素及其子元素。

<div *ngIf="hero" >{{hero.name}}</div>

會被解析為

<ng-template [ngIf]="hero">
 <div>{{hero.name}}</div>
</ng-template>`

看下ngIf源碼

import {Directive, EmbeddedViewRef, Input, TemplateRef, ViewContainerRef} from '@angular/core';
@Directive({selector: '[ngIf]'})
export class NgIf {
 private _context: NgIfContext = new NgIfContext();
 private _thenTemplateRef: TemplateRef<NgIfContext>|null = null;
 private _elseTemplateRef: TemplateRef<NgIfContext>|null = null;
 private _thenViewRef: EmbeddedViewRef<NgIfContext>|null = null;
 private _elseViewRef: EmbeddedViewRef<NgIfContext>|null = null;

 constructor(private _viewContainer: ViewContainerRef, templateRef: TemplateRef<NgIfContext>) {
  this._thenTemplateRef = templateRef;
 }

 @Input()
 set ngIf(condition: any) {
  this._context.$implicit = this._context.ngIf = condition;
  this._updateView();
 }

 @Input()
 set ngIfThen(templateRef: TemplateRef<NgIfContext>) {
  this._thenTemplateRef = templateRef;
  this._thenViewRef = null; // clear previous view if any.
  this._updateView();
 }

 @Input()
 set ngIfElse(templateRef: TemplateRef<NgIfContext>) {
  this._elseTemplateRef = templateRef;
  this._elseViewRef = null; // clear previous view if any.
  this._updateView();
 }

 private _updateView() {
  if (this._context.$implicit) {
   if (!this._thenViewRef) {
    this._viewContainer.clear();
    this._elseViewRef = null;
    if (this._thenTemplateRef) {
     this._thenViewRef =
       this._viewContainer.createEmbeddedView(this._thenTemplateRef, this._context);
    }
   }
  } else {
   if (!this._elseViewRef) {
    this._viewContainer.clear();
    this._thenViewRef = null;
    if (this._elseTemplateRef) {
     this._elseViewRef =
       this._viewContainer.createEmbeddedView(this._elseTemplateRef, this._context);
    }
   }
  }
 }
}

export class NgIfContext {
 public $implicit: any = null;
 public ngIf: any = null;
}

ngIf的源碼并不難,它的核心就在于_updateView函數,它主要通過ViewContainerRef的createEmbeddedView和clear方法來實現模板TemplateRef的呈現和清除(先不關注當中的then和else等的具體實現)。它使用TemplateRef取得<ng-template>的內容,并通過ViewContainerRef來訪問這個視圖容器。

TemplateRef

TemplateRef 實例用于表示模板對象,TemplateRef 抽象類的定義如下:

 abstract get elementRef(): ElementRef;
 abstract createEmbeddedView(context: C): EmbeddedViewRef<C>;
}

在指令中通過依賴注入TemplateRef可以直接拿到ng-tempalte的TemplateRef,但是在component組件中我們則需要使用viewChild

<ng-template #tptest>
 <span>template test</span>
</ng-template>

@ViewChild('tptest') tptest: TemplateRef<any>;

ViewContainerRef

ViewContainerRef 實例提供了 createEmbeddedView() 方法,該方法接收 TemplateRef 對象作為參數,并將模板中的內容作為容器 (comment 元素) 的兄弟元素,插入到頁面中。

export abstract class ViewContainerRef {
  /*基于TemplateRef對象創建Embedded View(內嵌視圖),然后根據`index`指定的值,插入到容器中。 
  如果沒有指定`index`的值,新創建的視圖將作為容器中的最后一個視圖插入。*/ 
 abstract createEmbeddedView<C>(
   templateRef: TemplateRef<C>, //內嵌視圖
   context?: C, index?: number): // 創建上下文
   EmbeddedViewRef<C>;
}

createEmbeddedView:context

創建Template 自身 Context 的屬性,以ngFor為例:
查看ngFor Context源碼:

export class NgForOfContext<T> {
 constructor(
   public $implicit: T, public ngForOf: NgIterable<T>, public index: number,
   public count: number) {}

 get first(): boolean { return this.index === 0; }

 get last(): boolean { return this.index === this.count - 1; }

 get even(): boolean { return this.index % 2 === 0; }

 get odd(): boolean { return !this.even; }
}

<div *ngFor="let hero of heroes; let i=index; let odd=odd">
 ({{i}}) {{hero.name}}
</div>


解析后:

<ng-template ngFor let-hero [ngForOf]="heroes" let-i="index" let-odd="odd" >
 <div>({{i}}) {{hero.name}}</div>
</ng-template>

從例子中可以看到,通過let-i let-odd可以獲取到Template的context,這是angular提供的一種語法。因為在 Angular中是沒有作用域繼承的,所以在模版中無法隱式實現兩個無關數據源。一個簡單的實現方案就是:一個顯式、一個隱式。由于ng-template tag 是寫在某個 Component 的 template屬性中的,所以在 ng-template tag 之下的部分當然能訪問的也只有 Component 作為 Context 提供的屬性,從而保持行為的一致性,而如果需要訪問到 Template 的 Context,我們就需要使用額外的引入語法。比如 let-i="index",就是把 Template Context 中的 index屬性引入到當前的 Component Context 中并賦予別名 i,這樣,我們就能夠使用 i 這個標識符來訪問到 Template Context 中的屬性了,并且仍然保持了行為的一致性和作用域的獨立性。

模板輸入變量是這樣一種變量,你可以在單個實例的模板中引用它的值。 這個例子中有好幾個模板輸入變量:hero、i和odd。 它們都是用let作為前導關鍵字。

模板輸入變量和模板引用變量是不同的,無論是在語義上還是語法上。

我們使用let關鍵字(如let hero)在模板中聲明一個模板輸入變量。 這個變量的范圍被限制在所重復模板的單一實例上。
而聲明模板引用變量使用的是給變量名加#前綴的方式(#var)。 一個引用變量引用的是它所附著到的元素、組件或指令。它可以在整個模板任意位置**訪問。

模板輸入變量和引用變量具有各自獨立的命名空間。let hero中的hero和#hero中的hero并不是同一個變量。

總結:

<ng-template>在ng中主要通過viewChild TemplateRef ViewContainerRef來實現結構性操作。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

大城县| 黄陵县| 洪湖市| 洛宁县| 蒲江县| 建宁县| 定兴县| 永福县| 缙云县| 永仁县| 平乡县| 无棣县| 镇康县| 墨江| 陈巴尔虎旗| 富平县| 肇州县| 昭通市| 桦川县| 驻马店市| 四川省| 绥德县| 金川县| 麻栗坡县| 高雄县| 台中市| 易门县| 洱源县| 桃源县| 四子王旗| 于都县| 兴国县| 舒兰市| 横山县| 金门县| 博客| 桂平市| 法库县| 林芝县| 乾安县| 新兴县|