實作虛擬捲動功能,讓大型清單的回應速度更快。
捲動清單是目前最常見的 UI 模式之一,無論是在您喜愛的社群媒體網站上瀏覽無限捲動的動態消息,或是瀏覽企業資訊主頁,都會用到捲動清單。當捲動清單變得非常長 (數百、數千或數十萬個項目) 時,應用程式效能可能會受到影響。
由於應用程式必須先載入並轉譯所有資料,因此大型清單的載入速度可能較慢。由於清單中的每個項目可能含有豐富的資料、媒體和功能,因此轉譯和瀏覽的速度也可能較慢。
使用者在載入或捲動網頁時可能會遇到問題,導致他們感到不耐煩並放棄網頁。
使用元件開發套件在 Angular 中進行虛擬捲動
虛擬捲動是解決這些縮放問題的主要技術。虛擬捲動功能會提供適當大小的捲動列,讓使用者有瀏覽大型清單的感受,並且能夠瀏覽清單,而不需要應用程式將整個清單保留在記憶體中,或在頁面上算繪清單。
在 Angular 中,虛擬捲動功能是由 元件開發套件 (CDK) 提供。只要修改列舉清單的方式,並提供幾個額外的設定參數,CDK 的虛擬捲動功能就會自動管理清單的虛擬轉譯,進而提升網頁效能和回應速度。
系統不會一次轉譯整個清單,而是只會轉譯可在螢幕上顯示的項目子集 (加上一個小緩衝區)。使用者瀏覽時,系統會計算及轉譯新的項目子集,並視需要重複使用現有的 DOM。
本篇文章的其餘部分將逐步說明如何設定基本虛擬捲動功能。您可以在這個範例應用程式中查看完整的工作範例:
設定虛擬捲動功能
首先,請確認您已使用偏好的套件管理工具安裝 @angular/cdk
。如要使用 npm 安裝,請在終端機中執行下列指令:
npm install --save @angular/cdk
在應用程式中新增 ScrollingModule
安裝 CDK 後,請從 @angular/cdk/scrolling
套件匯入 ScrollingModule
,以便處理虛擬捲動功能。然後將其加入模組的匯入陣列:
import {ScrollingModule} from '@angular/cdk/scrolling';
...
imports: [
ScrollingModule
...
]
...
建立可視區域
如要瞭解套件運作方式,請嘗試建立元件,其中包含從 0 到 99,999 的簡單數字清單:
@Component({
template: `<div *ngFor="let item of list">{{item}}</div>`
})
export class ScrollComponent {
list = Array.from({length: 100000}).map((_, i) => i);
}
瀏覽器算繪應用程式時,必須算繪 100,000 個個別 <div>
元素。這對簡單的文字節點來說或許沒什麼問題,但重複範本中的任何複雜性都無法順利擴充,而且任何事件監聽器都會大幅增加。
如要新增虛擬捲動功能並避免上述問題,您必須在 <cdk-virtual-scroll-viewport>
元素中包裝清單,藉此建立可視區域:
@Component({
template: `<cdk-virtual-scroll-viewport>
<div *ngFor="let item of list">{{item}}</div>
</cdk-virtual-scroll-viewport>`
})
export class ScrollComponent {
list = Array.from({length: 100000}).map((_, i) => i);
}
由於 ScrollingModule
會動態轉譯清單的子集,因此您必須透過標準 CSS 指定可視區域的高度。您也必須指定 itemSize
,為視區提供內容提示。模組會利用這些資訊,判斷在特定時間點應保留多少項目,以及如何顯示適當大小的捲軸。
@Component({
template: `<cdk-virtual-scroll-viewport itemSize="18" style="height:80vh">
<div *ngFor="let item of list">{{item}}</div>
</cdk-virtual-scroll-viewport>`
})
export class ScrollComponent {
list = Array.from({length: 100000}).map((_, i) => i);
}
最後,將 *ngFor
轉換為 *cdkVirtualFor
:
@Component({
template: `<cdk-virtual-scroll-viewport itemSize="18" style="height:80vh">
<div *cdkVirtualFor="let item of list">{{item}}</div>
</cdk-virtual-scroll-viewport>`
})
export class ScrollComponent {
list = Array.from({length: 100000}).map((_, i) => i);
}
檢視區不會逐一檢視整個清單,而是會動態識別並逐一檢視使用者清單的正確子集。如今,當使用者載入網頁時,CDK 應會轉譯適合顯示在畫面上的清單子集 (加上一點緩衝區),而檢視區中的任何捲動事件都會載入並轉譯清單的適當子集:
進一步
CDK 的虛擬捲動功能遠遠超出這個基本範例。在範例應用程式中,整個清單都位於記憶體中,但系統可視需要擷取清單,以便用於更複雜的應用程式。如要進一步瞭解 ScrollingModule
和 cdkVirtualOf
指令的其他功能,請參閱 CDK 說明文件中的 Scrolling
相關資訊。
主頁橫幅圖片由 Mr Cup / Fabien Barral 提供,來源:Unsplash。