Skip to content

Commit 42ed989

Browse files
committed
fix(cdk/scrolling): virtual list not updating when source array is mutated
When an array is passed to `cdkVirtualFor`, it falls back to creating an `ArrayDataSource` which won't emit if the array has changed. Since the intention of the `cdkVirtualFor` is to be (more or less) a drop-in alternative for `ngFor`, it is expected that it'll update if the data has changed. These changes add a new type of data source that will allow us to detect changes on the data and to update the view. Fixes #14635. Fixes #14501.
1 parent aa7dc00 commit 42ed989

File tree

6 files changed

+108
-4
lines changed

6 files changed

+108
-4
lines changed

src/cdk/collections/array-data-source.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ import {Observable, isObservable, of as observableOf} from 'rxjs';
1010
import {DataSource} from './data-source';
1111

1212

13-
/** DataSource wrapper for a native array. */
13+
/** DataSource wrapper for a static native array. */
1414
export class ArrayDataSource<T> extends DataSource<T> {
1515
constructor(private _data: T[] | ReadonlyArray<T> | Observable<T[] | ReadonlyArray<T>>) {
1616
super();
+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
/**
2+
* @license
3+
* Copyright Google LLC All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
import {DataSource} from './data-source';
10+
import {IterableDiffer, IterableDiffers, TrackByFunction} from '@angular/core';
11+
import {Subject, Observable} from 'rxjs';
12+
13+
/** DataSource wrapper for an iterable whose value might change. Emits when changes are detected. */
14+
export class DifferDataSource<T> extends DataSource<T> {
15+
private _differ: IterableDiffer<T>;
16+
private _changes = new Subject<T[]>();
17+
18+
constructor(
19+
private _differs: IterableDiffers,
20+
private _iterable: T[],
21+
trackBy?: TrackByFunction<T>) {
22+
23+
super();
24+
this._differ = _differs.find(_iterable).create(trackBy);
25+
}
26+
27+
connect(): Observable<T[] | ReadonlyArray<T>> {
28+
return this._changes;
29+
}
30+
31+
disconnect() {
32+
this._changes.complete();
33+
}
34+
35+
/** Checks the array for changes. */
36+
doCheck() {
37+
if (this._differ.diff(this._iterable)) {
38+
this._changes.next(this._iterable);
39+
}
40+
}
41+
42+
/** Switches the `trackBy` function of the data source. */
43+
switchTrackBy(trackBy?: TrackByFunction<T>) {
44+
this._differ = this._differs.find(this._iterable).create(trackBy);
45+
this.doCheck();
46+
}
47+
}

src/cdk/collections/public-api.ts

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
*/
88

99
export * from './array-data-source';
10+
export * from './differ-data-source';
1011
export * from './collection-viewer';
1112
export * from './data-source';
1213
export * from './dispose-view-repeater-strategy';

src/cdk/scrolling/virtual-for-of.ts

+29-3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import {
1010
ArrayDataSource,
1111
CollectionViewer,
1212
DataSource,
13+
DifferDataSource,
1314
ListRange,
1415
isDataSource,
1516
_RecycleViewRepeaterStrategy,
@@ -95,20 +96,33 @@ export class CdkVirtualForOf<T> implements
9596
/** Subject that emits when a new DataSource instance is given. */
9697
private _dataSourceChanges = new Subject<DataSource<T>>();
9798

99+
/**
100+
* Current differ data source. Needs to be kept in a separate
101+
* property so we can run change detection on it.
102+
*/
103+
private _differDataSource: DifferDataSource<T> | null = null;
104+
98105
/** The DataSource to display. */
99106
@Input()
100107
get cdkVirtualForOf(): DataSource<T> | Observable<T[]> | NgIterable<T> | null | undefined {
101108
return this._cdkVirtualForOf;
102109
}
103110
set cdkVirtualForOf(value: DataSource<T> | Observable<T[]> | NgIterable<T> | null | undefined) {
104111
this._cdkVirtualForOf = value;
112+
113+
let dataSource: DataSource<T>;
114+
105115
if (isDataSource(value)) {
106-
this._dataSourceChanges.next(value);
116+
dataSource = value;
117+
} else if (Array.isArray(value)) {
118+
this._differDataSource = dataSource =
119+
new DifferDataSource(this._differs, value, this.cdkVirtualForTrackBy);
107120
} else {
108121
// If value is an an NgIterable, convert it to an array.
109-
this._dataSourceChanges.next(new ArrayDataSource<T>(
110-
isObservable(value) ? value : Array.from(value || [])));
122+
dataSource = new ArrayDataSource<T>(isObservable(value) ? value : Array.from(value || []));
111123
}
124+
125+
this._dataSourceChanges.next(dataSource);
112126
}
113127

114128
_cdkVirtualForOf: DataSource<T> | Observable<T[]> | NgIterable<T> | null | undefined;
@@ -126,6 +140,10 @@ export class CdkVirtualForOf<T> implements
126140
this._cdkVirtualForTrackBy = fn ?
127141
(index, item) => fn(index + (this._renderedRange ? this._renderedRange.start : 0), item) :
128142
undefined;
143+
144+
if (this._differDataSource) {
145+
this._differDataSource.switchTrackBy(this._cdkVirtualForTrackBy);
146+
}
129147
}
130148
private _cdkVirtualForTrackBy: TrackByFunction<T> | undefined;
131149

@@ -255,6 +273,10 @@ export class CdkVirtualForOf<T> implements
255273
}
256274

257275
ngDoCheck() {
276+
if (this._differDataSource) {
277+
this._differDataSource.doCheck();
278+
}
279+
258280
if (this._differ && this._needsUpdate) {
259281
// TODO(mmalerba): We should differentiate needs update due to scrolling and a new portion of
260282
// this list being rendered (can use simpler algorithm) vs needs update due to data actually
@@ -303,6 +325,10 @@ export class CdkVirtualForOf<T> implements
303325

304326
if (oldDs) {
305327
oldDs.disconnect(this);
328+
329+
if (oldDs === this._differDataSource) {
330+
this._differDataSource = null;
331+
}
306332
}
307333

308334
this._needsUpdate = true;

src/cdk/scrolling/virtual-scroll-viewport.spec.ts

+22
Original file line numberDiff line numberDiff line change
@@ -574,6 +574,28 @@ describe('CdkVirtualScrollViewport', () => {
574574
expect(dataSource.disconnect).toHaveBeenCalled();
575575
}));
576576

577+
it('should work if the iterable is mutated', fakeAsync(() => {
578+
testComponent.items = [];
579+
finishInit(fixture);
580+
581+
expect(viewport.getRenderedRange())
582+
.toEqual({start: 0, end: 0}, 'no items should be rendered');
583+
584+
testComponent.items.push(1, 2, 3);
585+
fixture.detectChanges();
586+
flush();
587+
588+
expect(viewport.getRenderedRange())
589+
.toEqual({start: 0, end: 3}, 'newly emitted items should be rendered');
590+
591+
testComponent.items.pop();
592+
fixture.detectChanges();
593+
flush();
594+
595+
expect(viewport.getRenderedRange())
596+
.toEqual({start: 0, end: 2}, 'last item to be removed');
597+
}));
598+
577599
it('should trackBy value by default', fakeAsync(() => {
578600
testComponent.items = [];
579601
spyOn(testComponent.virtualForOf._viewContainerRef, 'detach').and.callThrough();

tools/public_api_guard/cdk/collections.d.ts

+8
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,14 @@ export declare abstract class DataSource<T> {
6060
abstract disconnect(collectionViewer: CollectionViewer): void;
6161
}
6262

63+
export declare class DifferDataSource<T> extends DataSource<T> {
64+
constructor(_differs: IterableDiffers, _iterable: T[], trackBy?: TrackByFunction<T>);
65+
connect(): Observable<T[] | ReadonlyArray<T>>;
66+
disconnect(): void;
67+
doCheck(): void;
68+
switchTrackBy(trackBy?: TrackByFunction<T>): void;
69+
}
70+
6371
export declare function getMultipleValuesInSingleSelectionError(): Error;
6472

6573
export declare function isDataSource(value: any): value is DataSource<any>;

0 commit comments

Comments
 (0)