仮想スクロールを実装して、大きなリストの応答性を高めます。
スクロール リストは、お気に入りのソーシャル メディア サイトで無限スクロールのフィードを閲覧する場合でも、企業向けダッシュボードを操作する場合でも、今日最も一般的な UI パターンの一つです。スクロール リストが非常に長い場合(数百、数千、数十万のアイテムなど)、アプリのパフォーマンスが低下する可能性があります。
リストが大きい場合、アプリケーションはすべてのデータを事前に読み込んでレンダリングする必要があるため、読み込みに時間がかかることがあります。また、リスト内の各アイテムはリッチデータ、メディア、機能を使用できるため、レンダリングやナビゲーションが遅くなる可能性もあります。
ページの読み込みやスクロール時に問題が発生し、ユーザーの不満を招いてページから離脱する可能性があります。
Component Dev Kit を使用した Angular の仮想スクロール
このようなスケーリングの問題に対処するために使用される主な手法は、仮想スクロールです。仮想スクロールでは、適切なサイズのスクロールバーを表示することで、非常に大きなリストのように見せることができます。また、リスト全体をメモリに保持したり、ページにレンダリングしたりすることなく、リストを操作できます。
Angular では、コンポーネント開発キット(CDK)によって仮想スクロールが提供されます。リストを反復処理する方法を変更し、いくつかの追加設定パラメータを指定することで、CDK の仮想スクロールがリストの仮想レンダリングを自動的に管理し、ページのパフォーマンスと応答性を向上させます。
リスト全体を一度にレンダリングするのではなく、画面に収まるアイテムのサブセット(および小さなバッファ)のみがレンダリングされます。ユーザーが移動すると、アイテムの新しいサブセットが計算されてレンダリングされ、必要に応じて既存の DOM が再利用されます。
この投稿の残りの部分では、基本的な仮想スクロールを設定する方法について説明します。動作する完全なサンプルは、こちらのサンプルアプリで確認できます。
仮想スクロールを設定する
まず、任意のパッケージ マネージャーを使用して @angular/cdk
がインストールされていることを確認します。npm を使用してインストールするには、ターミナルで次のコマンドを実行します。
npm install --save @angular/cdk
ScrollingModule
をアプリに追加します。
CDK がインストールされている状態で、仮想スクロールを処理する ScrollingModule
を @angular/cdk/scrolling
パッケージからインポートします。次に、これをモジュールの imports 配列に追加します。
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);
}
ブラウザがアプリをレンダリングする際に、10 万個の個々の <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
を指定して、ビューポートにコンテンツに関するヒントを与える必要があります。このモジュールは、この情報を使用して、特定の時点で DOM に保持するアイテムの数と、適切なサイズのスクロールバーをレンダリングする方法を決定します。
@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 より。