diff --git a/src/cdk/collections/array-data-source.ts b/src/cdk/collections/array-data-source.ts index 04c2957926c4..7f9c8a195376 100644 --- a/src/cdk/collections/array-data-source.ts +++ b/src/cdk/collections/array-data-source.ts @@ -10,7 +10,7 @@ import {Observable, isObservable, of as observableOf} from 'rxjs'; import {DataSource} from './data-source'; -/** DataSource wrapper for a native array. */ +/** DataSource wrapper for a static native array. */ export class ArrayDataSource extends DataSource { constructor(private _data: T[] | ReadonlyArray | Observable>) { super(); diff --git a/src/cdk/collections/differ-data-source.ts b/src/cdk/collections/differ-data-source.ts new file mode 100644 index 000000000000..642b09dd0380 --- /dev/null +++ b/src/cdk/collections/differ-data-source.ts @@ -0,0 +1,47 @@ +/** + * @license + * Copyright Google LLC All Rights Reserved. + * + * Use of this source code is governed by an MIT-style license that can be + * found in the LICENSE file at https://angular.io/license + */ + +import {DataSource} from './data-source'; +import {IterableDiffer, IterableDiffers, TrackByFunction} from '@angular/core'; +import {Subject, Observable} from 'rxjs'; + +/** DataSource wrapper for an iterable whose value might change. Emits when changes are detected. */ +export class DifferDataSource extends DataSource { + private _differ: IterableDiffer; + private _changes = new Subject(); + + constructor( + private _differs: IterableDiffers, + private _iterable: T[], + trackBy?: TrackByFunction) { + + super(); + this._differ = _differs.find(_iterable).create(trackBy); + } + + connect(): Observable> { + return this._changes; + } + + disconnect() { + this._changes.complete(); + } + + /** Checks the array for changes. */ + doCheck() { + if (this._differ.diff(this._iterable)) { + this._changes.next(this._iterable); + } + } + + /** Switches the `trackBy` function of the data source. */ + switchTrackBy(trackBy?: TrackByFunction) { + this._differ = this._differs.find(this._iterable).create(trackBy); + this.doCheck(); + } +} diff --git a/src/cdk/collections/public-api.ts b/src/cdk/collections/public-api.ts index 2d2d06f1a2eb..87e0515bf837 100644 --- a/src/cdk/collections/public-api.ts +++ b/src/cdk/collections/public-api.ts @@ -7,6 +7,7 @@ */ export * from './array-data-source'; +export * from './differ-data-source'; export * from './collection-viewer'; export * from './data-source'; export * from './dispose-view-repeater-strategy'; diff --git a/src/cdk/scrolling/virtual-for-of.ts b/src/cdk/scrolling/virtual-for-of.ts index 9b2aff448f62..49f14c7db750 100644 --- a/src/cdk/scrolling/virtual-for-of.ts +++ b/src/cdk/scrolling/virtual-for-of.ts @@ -10,6 +10,7 @@ import { ArrayDataSource, CollectionViewer, DataSource, + DifferDataSource, ListRange, isDataSource, _RecycleViewRepeaterStrategy, @@ -95,6 +96,12 @@ export class CdkVirtualForOf implements /** Subject that emits when a new DataSource instance is given. */ private _dataSourceChanges = new Subject>(); + /** + * Current differ data source. Needs to be kept in a separate + * property so we can run change detection on it. + */ + private _differDataSource: DifferDataSource | null = null; + /** The DataSource to display. */ @Input() get cdkVirtualForOf(): DataSource | Observable | NgIterable | null | undefined { @@ -102,13 +109,20 @@ export class CdkVirtualForOf implements } set cdkVirtualForOf(value: DataSource | Observable | NgIterable | null | undefined) { this._cdkVirtualForOf = value; + + let dataSource: DataSource; + if (isDataSource(value)) { - this._dataSourceChanges.next(value); + dataSource = value; + } else if (Array.isArray(value)) { + this._differDataSource = dataSource = + new DifferDataSource(this._differs, value, this.cdkVirtualForTrackBy); } else { // If value is an an NgIterable, convert it to an array. - this._dataSourceChanges.next(new ArrayDataSource( - isObservable(value) ? value : Array.from(value || []))); + dataSource = new ArrayDataSource(isObservable(value) ? value : Array.from(value || [])); } + + this._dataSourceChanges.next(dataSource); } _cdkVirtualForOf: DataSource | Observable | NgIterable | null | undefined; @@ -126,6 +140,10 @@ export class CdkVirtualForOf implements this._cdkVirtualForTrackBy = fn ? (index, item) => fn(index + (this._renderedRange ? this._renderedRange.start : 0), item) : undefined; + + if (this._differDataSource) { + this._differDataSource.switchTrackBy(this._cdkVirtualForTrackBy); + } } private _cdkVirtualForTrackBy: TrackByFunction | undefined; @@ -255,6 +273,10 @@ export class CdkVirtualForOf implements } ngDoCheck() { + if (this._differDataSource) { + this._differDataSource.doCheck(); + } + if (this._differ && this._needsUpdate) { // TODO(mmalerba): We should differentiate needs update due to scrolling and a new portion of // this list being rendered (can use simpler algorithm) vs needs update due to data actually @@ -303,6 +325,10 @@ export class CdkVirtualForOf implements if (oldDs) { oldDs.disconnect(this); + + if (oldDs === this._differDataSource) { + this._differDataSource = null; + } } this._needsUpdate = true; diff --git a/src/cdk/scrolling/virtual-scroll-viewport.spec.ts b/src/cdk/scrolling/virtual-scroll-viewport.spec.ts index 2ac1602f2a8b..466e7924d4bd 100644 --- a/src/cdk/scrolling/virtual-scroll-viewport.spec.ts +++ b/src/cdk/scrolling/virtual-scroll-viewport.spec.ts @@ -574,6 +574,28 @@ describe('CdkVirtualScrollViewport', () => { expect(dataSource.disconnect).toHaveBeenCalled(); })); + it('should work if the iterable is mutated', fakeAsync(() => { + testComponent.items = []; + finishInit(fixture); + + expect(viewport.getRenderedRange()) + .toEqual({start: 0, end: 0}, 'no items should be rendered'); + + testComponent.items.push(1, 2, 3); + fixture.detectChanges(); + flush(); + + expect(viewport.getRenderedRange()) + .toEqual({start: 0, end: 3}, 'newly emitted items should be rendered'); + + testComponent.items.pop(); + fixture.detectChanges(); + flush(); + + expect(viewport.getRenderedRange()) + .toEqual({start: 0, end: 2}, 'last item to be removed'); + })); + it('should trackBy value by default', fakeAsync(() => { testComponent.items = []; spyOn(testComponent.virtualForOf._viewContainerRef, 'detach').and.callThrough(); diff --git a/tools/public_api_guard/cdk/collections.d.ts b/tools/public_api_guard/cdk/collections.d.ts index 8feb88fc72e6..cac67e2a58cc 100644 --- a/tools/public_api_guard/cdk/collections.d.ts +++ b/tools/public_api_guard/cdk/collections.d.ts @@ -60,6 +60,14 @@ export declare abstract class DataSource { abstract disconnect(collectionViewer: CollectionViewer): void; } +export declare class DifferDataSource extends DataSource { + constructor(_differs: IterableDiffers, _iterable: T[], trackBy?: TrackByFunction); + connect(): Observable>; + disconnect(): void; + doCheck(): void; + switchTrackBy(trackBy?: TrackByFunction): void; +} + export declare function getMultipleValuesInSingleSelectionError(): Error; export declare function isDataSource(value: any): value is DataSource;