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 | import { Directionality } from '@angular/cdk/bidi';
|
9 | import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, inject, Inject, Input, NgZone, Optional, Output, ViewChild, ViewEncapsulation, } from '@angular/core';
|
10 | import { Platform } from '@angular/cdk/platform';
|
11 | import { animationFrameScheduler, asapScheduler, Observable, Subject, Subscription, } from 'rxjs';
|
12 | import { auditTime, startWith, takeUntil } from 'rxjs/operators';
|
13 | import { ScrollDispatcher } from './scroll-dispatcher';
|
14 | import { CdkScrollable } from './scrollable';
|
15 | import { VIRTUAL_SCROLL_STRATEGY } from './virtual-scroll-strategy';
|
16 | import { ViewportRuler } from './viewport-ruler';
|
17 | import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
18 | import { CdkVirtualScrollable, VIRTUAL_SCROLLABLE } from './virtual-scrollable';
|
19 | import * as i0 from "@angular/core";
|
20 | import * as i1 from "@angular/cdk/bidi";
|
21 | import * as i2 from "./scroll-dispatcher";
|
22 | import * as i3 from "./viewport-ruler";
|
23 | import * as i4 from "./virtual-scrollable";
|
24 | /** Checks if the given ranges are equal. */
|
25 | function rangesEqual(r1, r2) {
|
26 | return r1.start == r2.start && r1.end == r2.end;
|
27 | }
|
28 | /**
|
29 | * Scheduler to be used for scroll events. Needs to fall back to
|
30 | * something that doesn't rely on requestAnimationFrame on environments
|
31 | * that don't support it (e.g. server-side rendering).
|
32 | */
|
33 | const SCROLL_SCHEDULER = typeof requestAnimationFrame !== 'undefined' ? animationFrameScheduler : asapScheduler;
|
34 | /** A viewport that virtualizes its scrolling with the help of `CdkVirtualForOf`. */
|
35 | class CdkVirtualScrollViewport extends CdkVirtualScrollable {
|
36 | /** The direction the viewport scrolls. */
|
37 | get orientation() {
|
38 | return this._orientation;
|
39 | }
|
40 | set orientation(orientation) {
|
41 | if (this._orientation !== orientation) {
|
42 | this._orientation = orientation;
|
43 | this._calculateSpacerSize();
|
44 | }
|
45 | }
|
46 | /**
|
47 | * Whether rendered items should persist in the DOM after scrolling out of view. By default, items
|
48 | * will be removed.
|
49 | */
|
50 | get appendOnly() {
|
51 | return this._appendOnly;
|
52 | }
|
53 | set appendOnly(value) {
|
54 | this._appendOnly = coerceBooleanProperty(value);
|
55 | }
|
56 | constructor(elementRef, _changeDetectorRef, ngZone, _scrollStrategy, dir, scrollDispatcher, viewportRuler, scrollable) {
|
57 | super(elementRef, scrollDispatcher, ngZone, dir);
|
58 | this.elementRef = elementRef;
|
59 | this._changeDetectorRef = _changeDetectorRef;
|
60 | this._scrollStrategy = _scrollStrategy;
|
61 | this.scrollable = scrollable;
|
62 | this._platform = inject(Platform);
|
63 | /** Emits when the viewport is detached from a CdkVirtualForOf. */
|
64 | this._detachedSubject = new Subject();
|
65 | /** Emits when the rendered range changes. */
|
66 | this._renderedRangeSubject = new Subject();
|
67 | this._orientation = 'vertical';
|
68 | this._appendOnly = false;
|
69 | // Note: we don't use the typical EventEmitter here because we need to subscribe to the scroll
|
70 | // strategy lazily (i.e. only if the user is actually listening to the events). We do this because
|
71 | // depending on how the strategy calculates the scrolled index, it may come at a cost to
|
72 | // performance.
|
73 | /** Emits when the index of the first element visible in the viewport changes. */
|
74 | this.scrolledIndexChange = new Observable((observer) => this._scrollStrategy.scrolledIndexChange.subscribe(index => Promise.resolve().then(() => this.ngZone.run(() => observer.next(index)))));
|
75 | /** A stream that emits whenever the rendered range changes. */
|
76 | this.renderedRangeStream = this._renderedRangeSubject;
|
77 | /**
|
78 | * The total size of all content (in pixels), including content that is not currently rendered.
|
79 | */
|
80 | this._totalContentSize = 0;
|
81 | /** A string representing the `style.width` property value to be used for the spacer element. */
|
82 | this._totalContentWidth = '';
|
83 | /** A string representing the `style.height` property value to be used for the spacer element. */
|
84 | this._totalContentHeight = '';
|
85 | /** The currently rendered range of indices. */
|
86 | this._renderedRange = { start: 0, end: 0 };
|
87 | /** The length of the data bound to this viewport (in number of items). */
|
88 | this._dataLength = 0;
|
89 | /** The size of the viewport (in pixels). */
|
90 | this._viewportSize = 0;
|
91 | /** The last rendered content offset that was set. */
|
92 | this._renderedContentOffset = 0;
|
93 | /**
|
94 | * Whether the last rendered content offset was to the end of the content (and therefore needs to
|
95 | * be rewritten as an offset to the start of the content).
|
96 | */
|
97 | this._renderedContentOffsetNeedsRewrite = false;
|
98 | /** Whether there is a pending change detection cycle. */
|
99 | this._isChangeDetectionPending = false;
|
100 | /** A list of functions to run after the next change detection cycle. */
|
101 | this._runAfterChangeDetection = [];
|
102 | /** Subscription to changes in the viewport size. */
|
103 | this._viewportChanges = Subscription.EMPTY;
|
104 | if (!_scrollStrategy && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
105 | throw Error('Error: cdk-virtual-scroll-viewport requires the "itemSize" property to be set.');
|
106 | }
|
107 | this._viewportChanges = viewportRuler.change().subscribe(() => {
|
108 | this.checkViewportSize();
|
109 | });
|
110 | if (!this.scrollable) {
|
111 | // No scrollable is provided, so the virtual-scroll-viewport needs to become a scrollable
|
112 | this.elementRef.nativeElement.classList.add('cdk-virtual-scrollable');
|
113 | this.scrollable = this;
|
114 | }
|
115 | }
|
116 | ngOnInit() {
|
117 | // Scrolling depends on the element dimensions which we can't get during SSR.
|
118 | if (!this._platform.isBrowser) {
|
119 | return;
|
120 | }
|
121 | if (this.scrollable === this) {
|
122 | super.ngOnInit();
|
123 | }
|
124 | // It's still too early to measure the viewport at this point. Deferring with a promise allows
|
125 | // the Viewport to be rendered with the correct size before we measure. We run this outside the
|
126 | // zone to avoid causing more change detection cycles. We handle the change detection loop
|
127 | // ourselves instead.
|
128 | this.ngZone.runOutsideAngular(() => Promise.resolve().then(() => {
|
129 | this._measureViewportSize();
|
130 | this._scrollStrategy.attach(this);
|
131 | this.scrollable
|
132 | .elementScrolled()
|
133 | .pipe(
|
134 | // Start off with a fake scroll event so we properly detect our initial position.
|
135 | startWith(null),
|
136 | // Collect multiple events into one until the next animation frame. This way if
|
137 | // there are multiple scroll events in the same frame we only need to recheck
|
138 | // our layout once.
|
139 | auditTime(0, SCROLL_SCHEDULER))
|
140 | .subscribe(() => this._scrollStrategy.onContentScrolled());
|
141 | this._markChangeDetectionNeeded();
|
142 | }));
|
143 | }
|
144 | ngOnDestroy() {
|
145 | this.detach();
|
146 | this._scrollStrategy.detach();
|
147 | // Complete all subjects
|
148 | this._renderedRangeSubject.complete();
|
149 | this._detachedSubject.complete();
|
150 | this._viewportChanges.unsubscribe();
|
151 | super.ngOnDestroy();
|
152 | }
|
153 | /** Attaches a `CdkVirtualScrollRepeater` to this viewport. */
|
154 | attach(forOf) {
|
155 | if (this._forOf && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
156 | throw Error('CdkVirtualScrollViewport is already attached.');
|
157 | }
|
158 | // Subscribe to the data stream of the CdkVirtualForOf to keep track of when the data length
|
159 | // changes. Run outside the zone to avoid triggering change detection, since we're managing the
|
160 | // change detection loop ourselves.
|
161 | this.ngZone.runOutsideAngular(() => {
|
162 | this._forOf = forOf;
|
163 | this._forOf.dataStream.pipe(takeUntil(this._detachedSubject)).subscribe(data => {
|
164 | const newLength = data.length;
|
165 | if (newLength !== this._dataLength) {
|
166 | this._dataLength = newLength;
|
167 | this._scrollStrategy.onDataLengthChanged();
|
168 | }
|
169 | this._doChangeDetection();
|
170 | });
|
171 | });
|
172 | }
|
173 | /** Detaches the current `CdkVirtualForOf`. */
|
174 | detach() {
|
175 | this._forOf = null;
|
176 | this._detachedSubject.next();
|
177 | }
|
178 | /** Gets the length of the data bound to this viewport (in number of items). */
|
179 | getDataLength() {
|
180 | return this._dataLength;
|
181 | }
|
182 | /** Gets the size of the viewport (in pixels). */
|
183 | getViewportSize() {
|
184 | return this._viewportSize;
|
185 | }
|
186 | // TODO(mmalerba): This is technically out of sync with what's really rendered until a render
|
187 | // cycle happens. I'm being careful to only call it after the render cycle is complete and before
|
188 | // setting it to something else, but its error prone and should probably be split into
|
189 | // `pendingRange` and `renderedRange`, the latter reflecting whats actually in the DOM.
|
190 | /** Get the current rendered range of items. */
|
191 | getRenderedRange() {
|
192 | return this._renderedRange;
|
193 | }
|
194 | measureBoundingClientRectWithScrollOffset(from) {
|
195 | return this.getElementRef().nativeElement.getBoundingClientRect()[from];
|
196 | }
|
197 | /**
|
198 | * Sets the total size of all content (in pixels), including content that is not currently
|
199 | * rendered.
|
200 | */
|
201 | setTotalContentSize(size) {
|
202 | if (this._totalContentSize !== size) {
|
203 | this._totalContentSize = size;
|
204 | this._calculateSpacerSize();
|
205 | this._markChangeDetectionNeeded();
|
206 | }
|
207 | }
|
208 | /** Sets the currently rendered range of indices. */
|
209 | setRenderedRange(range) {
|
210 | if (!rangesEqual(this._renderedRange, range)) {
|
211 | if (this.appendOnly) {
|
212 | range = { start: 0, end: Math.max(this._renderedRange.end, range.end) };
|
213 | }
|
214 | this._renderedRangeSubject.next((this._renderedRange = range));
|
215 | this._markChangeDetectionNeeded(() => this._scrollStrategy.onContentRendered());
|
216 | }
|
217 | }
|
218 | /**
|
219 | * Gets the offset from the start of the viewport to the start of the rendered data (in pixels).
|
220 | */
|
221 | getOffsetToRenderedContentStart() {
|
222 | return this._renderedContentOffsetNeedsRewrite ? null : this._renderedContentOffset;
|
223 | }
|
224 | /**
|
225 | * Sets the offset from the start of the viewport to either the start or end of the rendered data
|
226 | * (in pixels).
|
227 | */
|
228 | setRenderedContentOffset(offset, to = 'to-start') {
|
229 | // In appendOnly, we always start from the top
|
230 | offset = this.appendOnly && to === 'to-start' ? 0 : offset;
|
231 | // For a horizontal viewport in a right-to-left language we need to translate along the x-axis
|
232 | // in the negative direction.
|
233 | const isRtl = this.dir && this.dir.value == 'rtl';
|
234 | const isHorizontal = this.orientation == 'horizontal';
|
235 | const axis = isHorizontal ? 'X' : 'Y';
|
236 | const axisDirection = isHorizontal && isRtl ? -1 : 1;
|
237 | let transform = `translate${axis}(${Number(axisDirection * offset)}px)`;
|
238 | this._renderedContentOffset = offset;
|
239 | if (to === 'to-end') {
|
240 | transform += ` translate${axis}(-100%)`;
|
241 | // The viewport should rewrite this as a `to-start` offset on the next render cycle. Otherwise
|
242 | // elements will appear to expand in the wrong direction (e.g. `mat-expansion-panel` would
|
243 | // expand upward).
|
244 | this._renderedContentOffsetNeedsRewrite = true;
|
245 | }
|
246 | if (this._renderedContentTransform != transform) {
|
247 | // We know this value is safe because we parse `offset` with `Number()` before passing it
|
248 | // into the string.
|
249 | this._renderedContentTransform = transform;
|
250 | this._markChangeDetectionNeeded(() => {
|
251 | if (this._renderedContentOffsetNeedsRewrite) {
|
252 | this._renderedContentOffset -= this.measureRenderedContentSize();
|
253 | this._renderedContentOffsetNeedsRewrite = false;
|
254 | this.setRenderedContentOffset(this._renderedContentOffset);
|
255 | }
|
256 | else {
|
257 | this._scrollStrategy.onRenderedOffsetChanged();
|
258 | }
|
259 | });
|
260 | }
|
261 | }
|
262 | /**
|
263 | * Scrolls to the given offset from the start of the viewport. Please note that this is not always
|
264 | * the same as setting `scrollTop` or `scrollLeft`. In a horizontal viewport with right-to-left
|
265 | * direction, this would be the equivalent of setting a fictional `scrollRight` property.
|
266 | * @param offset The offset to scroll to.
|
267 | * @param behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.
|
268 | */
|
269 | scrollToOffset(offset, behavior = 'auto') {
|
270 | const options = { behavior };
|
271 | if (this.orientation === 'horizontal') {
|
272 | options.start = offset;
|
273 | }
|
274 | else {
|
275 | options.top = offset;
|
276 | }
|
277 | this.scrollable.scrollTo(options);
|
278 | }
|
279 | /**
|
280 | * Scrolls to the offset for the given index.
|
281 | * @param index The index of the element to scroll to.
|
282 | * @param behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.
|
283 | */
|
284 | scrollToIndex(index, behavior = 'auto') {
|
285 | this._scrollStrategy.scrollToIndex(index, behavior);
|
286 | }
|
287 | /**
|
288 | * Gets the current scroll offset from the start of the scrollable (in pixels).
|
289 | * @param from The edge to measure the offset from. Defaults to 'top' in vertical mode and 'start'
|
290 | * in horizontal mode.
|
291 | */
|
292 | measureScrollOffset(from) {
|
293 | // This is to break the call cycle
|
294 | let measureScrollOffset;
|
295 | if (this.scrollable == this) {
|
296 | measureScrollOffset = (_from) => super.measureScrollOffset(_from);
|
297 | }
|
298 | else {
|
299 | measureScrollOffset = (_from) => this.scrollable.measureScrollOffset(_from);
|
300 | }
|
301 | return Math.max(0, measureScrollOffset(from ?? (this.orientation === 'horizontal' ? 'start' : 'top')) -
|
302 | this.measureViewportOffset());
|
303 | }
|
304 | /**
|
305 | * Measures the offset of the viewport from the scrolling container
|
306 | * @param from The edge to measure from.
|
307 | */
|
308 | measureViewportOffset(from) {
|
309 | let fromRect;
|
310 | const LEFT = 'left';
|
311 | const RIGHT = 'right';
|
312 | const isRtl = this.dir?.value == 'rtl';
|
313 | if (from == 'start') {
|
314 | fromRect = isRtl ? RIGHT : LEFT;
|
315 | }
|
316 | else if (from == 'end') {
|
317 | fromRect = isRtl ? LEFT : RIGHT;
|
318 | }
|
319 | else if (from) {
|
320 | fromRect = from;
|
321 | }
|
322 | else {
|
323 | fromRect = this.orientation === 'horizontal' ? 'left' : 'top';
|
324 | }
|
325 | const scrollerClientRect = this.scrollable.measureBoundingClientRectWithScrollOffset(fromRect);
|
326 | const viewportClientRect = this.elementRef.nativeElement.getBoundingClientRect()[fromRect];
|
327 | return viewportClientRect - scrollerClientRect;
|
328 | }
|
329 | /** Measure the combined size of all of the rendered items. */
|
330 | measureRenderedContentSize() {
|
331 | const contentEl = this._contentWrapper.nativeElement;
|
332 | return this.orientation === 'horizontal' ? contentEl.offsetWidth : contentEl.offsetHeight;
|
333 | }
|
334 | /**
|
335 | * Measure the total combined size of the given range. Throws if the range includes items that are
|
336 | * not rendered.
|
337 | */
|
338 | measureRangeSize(range) {
|
339 | if (!this._forOf) {
|
340 | return 0;
|
341 | }
|
342 | return this._forOf.measureRangeSize(range, this.orientation);
|
343 | }
|
344 | /** Update the viewport dimensions and re-render. */
|
345 | checkViewportSize() {
|
346 | // TODO: Cleanup later when add logic for handling content resize
|
347 | this._measureViewportSize();
|
348 | this._scrollStrategy.onDataLengthChanged();
|
349 | }
|
350 | /** Measure the viewport size. */
|
351 | _measureViewportSize() {
|
352 | this._viewportSize = this.scrollable.measureViewportSize(this.orientation);
|
353 | }
|
354 | /** Queue up change detection to run. */
|
355 | _markChangeDetectionNeeded(runAfter) {
|
356 | if (runAfter) {
|
357 | this._runAfterChangeDetection.push(runAfter);
|
358 | }
|
359 | // Use a Promise to batch together calls to `_doChangeDetection`. This way if we set a bunch of
|
360 | // properties sequentially we only have to run `_doChangeDetection` once at the end.
|
361 | if (!this._isChangeDetectionPending) {
|
362 | this._isChangeDetectionPending = true;
|
363 | this.ngZone.runOutsideAngular(() => Promise.resolve().then(() => {
|
364 | this._doChangeDetection();
|
365 | }));
|
366 | }
|
367 | }
|
368 | /** Run change detection. */
|
369 | _doChangeDetection() {
|
370 | this._isChangeDetectionPending = false;
|
371 | // Apply the content transform. The transform can't be set via an Angular binding because
|
372 | // bypassSecurityTrustStyle is banned in Google. However the value is safe, it's composed of
|
373 | // string literals, a variable that can only be 'X' or 'Y', and user input that is run through
|
374 | // the `Number` function first to coerce it to a numeric value.
|
375 | this._contentWrapper.nativeElement.style.transform = this._renderedContentTransform;
|
376 | // Apply changes to Angular bindings. Note: We must call `markForCheck` to run change detection
|
377 | // from the root, since the repeated items are content projected in. Calling `detectChanges`
|
378 | // instead does not properly check the projected content.
|
379 | this.ngZone.run(() => this._changeDetectorRef.markForCheck());
|
380 | const runAfterChangeDetection = this._runAfterChangeDetection;
|
381 | this._runAfterChangeDetection = [];
|
382 | for (const fn of runAfterChangeDetection) {
|
383 | fn();
|
384 | }
|
385 | }
|
386 | /** Calculates the `style.width` and `style.height` for the spacer element. */
|
387 | _calculateSpacerSize() {
|
388 | this._totalContentHeight =
|
389 | this.orientation === 'horizontal' ? '' : `${this._totalContentSize}px`;
|
390 | this._totalContentWidth =
|
391 | this.orientation === 'horizontal' ? `${this._totalContentSize}px` : '';
|
392 | }
|
393 | static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkVirtualScrollViewport, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i0.NgZone }, { token: VIRTUAL_SCROLL_STRATEGY, optional: true }, { token: i1.Directionality, optional: true }, { token: i2.ScrollDispatcher }, { token: i3.ViewportRuler }, { token: VIRTUAL_SCROLLABLE, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
394 | static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: CdkVirtualScrollViewport, isStandalone: true, selector: "cdk-virtual-scroll-viewport", inputs: { orientation: "orientation", appendOnly: "appendOnly" }, outputs: { scrolledIndexChange: "scrolledIndexChange" }, host: { properties: { "class.cdk-virtual-scroll-orientation-horizontal": "orientation === \"horizontal\"", "class.cdk-virtual-scroll-orientation-vertical": "orientation !== \"horizontal\"" }, classAttribute: "cdk-virtual-scroll-viewport" }, providers: [
|
395 | {
|
396 | provide: CdkScrollable,
|
397 | useFactory: (virtualScrollable, viewport) => virtualScrollable || viewport,
|
398 | deps: [[new Optional(), new Inject(VIRTUAL_SCROLLABLE)], CdkVirtualScrollViewport],
|
399 | },
|
400 | ], viewQueries: [{ propertyName: "_contentWrapper", first: true, predicate: ["contentWrapper"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<!--\n Wrap the rendered content in an element that will be used to offset it based on the scroll\n position.\n-->\n<div #contentWrapper class=\"cdk-virtual-scroll-content-wrapper\">\n <ng-content></ng-content>\n</div>\n<!--\n Spacer used to force the scrolling container to the correct size for the *total* number of items\n so that the scrollbar captures the size of the entire data set.\n-->\n<div class=\"cdk-virtual-scroll-spacer\"\n [style.width]=\"_totalContentWidth\" [style.height]=\"_totalContentHeight\"></div>\n", styles: ["cdk-virtual-scroll-viewport{display:block;position:relative;transform:translateZ(0)}.cdk-virtual-scrollable{overflow:auto;will-change:scroll-position;contain:strict;-webkit-overflow-scrolling:touch}.cdk-virtual-scroll-content-wrapper{position:absolute;top:0;left:0;contain:content}[dir=rtl] .cdk-virtual-scroll-content-wrapper{right:0;left:auto}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper{min-height:100%}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-left:0;padding-right:0;margin-left:0;margin-right:0;border-left-width:0;border-right-width:0;outline:none}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper{min-width:100%}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-top:0;padding-bottom:0;margin-top:0;margin-bottom:0;border-top-width:0;border-bottom-width:0;outline:none}.cdk-virtual-scroll-spacer{height:1px;transform-origin:0 0;flex:0 0 auto}[dir=rtl] .cdk-virtual-scroll-spacer{transform-origin:100% 0}"], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
401 | }
|
402 | export { CdkVirtualScrollViewport };
|
403 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkVirtualScrollViewport, decorators: [{
|
404 | type: Component,
|
405 | args: [{ selector: 'cdk-virtual-scroll-viewport', host: {
|
406 | 'class': 'cdk-virtual-scroll-viewport',
|
407 | '[class.cdk-virtual-scroll-orientation-horizontal]': 'orientation === "horizontal"',
|
408 | '[class.cdk-virtual-scroll-orientation-vertical]': 'orientation !== "horizontal"',
|
409 | }, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, providers: [
|
410 | {
|
411 | provide: CdkScrollable,
|
412 | useFactory: (virtualScrollable, viewport) => virtualScrollable || viewport,
|
413 | deps: [[new Optional(), new Inject(VIRTUAL_SCROLLABLE)], CdkVirtualScrollViewport],
|
414 | },
|
415 | ], template: "<!--\n Wrap the rendered content in an element that will be used to offset it based on the scroll\n position.\n-->\n<div #contentWrapper class=\"cdk-virtual-scroll-content-wrapper\">\n <ng-content></ng-content>\n</div>\n<!--\n Spacer used to force the scrolling container to the correct size for the *total* number of items\n so that the scrollbar captures the size of the entire data set.\n-->\n<div class=\"cdk-virtual-scroll-spacer\"\n [style.width]=\"_totalContentWidth\" [style.height]=\"_totalContentHeight\"></div>\n", styles: ["cdk-virtual-scroll-viewport{display:block;position:relative;transform:translateZ(0)}.cdk-virtual-scrollable{overflow:auto;will-change:scroll-position;contain:strict;-webkit-overflow-scrolling:touch}.cdk-virtual-scroll-content-wrapper{position:absolute;top:0;left:0;contain:content}[dir=rtl] .cdk-virtual-scroll-content-wrapper{right:0;left:auto}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper{min-height:100%}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-left:0;padding-right:0;margin-left:0;margin-right:0;border-left-width:0;border-right-width:0;outline:none}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper{min-width:100%}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-top:0;padding-bottom:0;margin-top:0;margin-bottom:0;border-top-width:0;border-bottom-width:0;outline:none}.cdk-virtual-scroll-spacer{height:1px;transform-origin:0 0;flex:0 0 auto}[dir=rtl] .cdk-virtual-scroll-spacer{transform-origin:100% 0}"] }]
|
416 | }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i0.NgZone }, { type: undefined, decorators: [{
|
417 | type: Optional
|
418 | }, {
|
419 | type: Inject,
|
420 | args: [VIRTUAL_SCROLL_STRATEGY]
|
421 | }] }, { type: i1.Directionality, decorators: [{
|
422 | type: Optional
|
423 | }] }, { type: i2.ScrollDispatcher }, { type: i3.ViewportRuler }, { type: i4.CdkVirtualScrollable, decorators: [{
|
424 | type: Optional
|
425 | }, {
|
426 | type: Inject,
|
427 | args: [VIRTUAL_SCROLLABLE]
|
428 | }] }]; }, propDecorators: { orientation: [{
|
429 | type: Input
|
430 | }], appendOnly: [{
|
431 | type: Input
|
432 | }], scrolledIndexChange: [{
|
433 | type: Output
|
434 | }], _contentWrapper: [{
|
435 | type: ViewChild,
|
436 | args: ['contentWrapper', { static: true }]
|
437 | }] } });
|
438 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-scroll-viewport.js","sourceRoot":"","sources":["../../../../../../src/cdk/scrolling/virtual-scroll-viewport.ts","../../../../../../src/cdk/scrolling/virtual-scroll-viewport.html"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {Directionality} from '@angular/cdk/bidi';\nimport {ListRange} from '@angular/cdk/collections';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  inject,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {Platform} from '@angular/cdk/platform';\nimport {\n  animationFrameScheduler,\n  asapScheduler,\n  Observable,\n  Subject,\n  Observer,\n  Subscription,\n} from 'rxjs';\nimport {auditTime, startWith, takeUntil} from 'rxjs/operators';\nimport {ScrollDispatcher} from './scroll-dispatcher';\nimport {CdkScrollable, ExtendedScrollToOptions} from './scrollable';\nimport {VIRTUAL_SCROLL_STRATEGY, VirtualScrollStrategy} from './virtual-scroll-strategy';\nimport {ViewportRuler} from './viewport-ruler';\nimport {CdkVirtualScrollRepeater} from './virtual-scroll-repeater';\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {CdkVirtualScrollable, VIRTUAL_SCROLLABLE} from './virtual-scrollable';\n\n/** Checks if the given ranges are equal. */\nfunction rangesEqual(r1: ListRange, r2: ListRange): boolean {\n  return r1.start == r2.start && r1.end == r2.end;\n}\n\n/**\n * Scheduler to be used for scroll events. Needs to fall back to\n * something that doesn't rely on requestAnimationFrame on environments\n * that don't support it (e.g. server-side rendering).\n */\nconst SCROLL_SCHEDULER =\n  typeof requestAnimationFrame !== 'undefined' ? animationFrameScheduler : asapScheduler;\n\n/** A viewport that virtualizes its scrolling with the help of `CdkVirtualForOf`. */\n@Component({\n  selector: 'cdk-virtual-scroll-viewport',\n  templateUrl: 'virtual-scroll-viewport.html',\n  styleUrls: ['virtual-scroll-viewport.css'],\n  host: {\n    'class': 'cdk-virtual-scroll-viewport',\n    '[class.cdk-virtual-scroll-orientation-horizontal]': 'orientation === \"horizontal\"',\n    '[class.cdk-virtual-scroll-orientation-vertical]': 'orientation !== \"horizontal\"',\n  },\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  providers: [\n    {\n      provide: CdkScrollable,\n      useFactory: (\n        virtualScrollable: CdkVirtualScrollable | null,\n        viewport: CdkVirtualScrollViewport,\n      ) => virtualScrollable || viewport,\n      deps: [[new Optional(), new Inject(VIRTUAL_SCROLLABLE)], CdkVirtualScrollViewport],\n    },\n  ],\n})\nexport class CdkVirtualScrollViewport extends CdkVirtualScrollable implements OnInit, OnDestroy {\n  private _platform = inject(Platform);\n\n  /** Emits when the viewport is detached from a CdkVirtualForOf. */\n  private readonly _detachedSubject = new Subject<void>();\n\n  /** Emits when the rendered range changes. */\n  private readonly _renderedRangeSubject = new Subject<ListRange>();\n\n  /** The direction the viewport scrolls. */\n  @Input()\n  get orientation() {\n    return this._orientation;\n  }\n\n  set orientation(orientation: 'horizontal' | 'vertical') {\n    if (this._orientation !== orientation) {\n      this._orientation = orientation;\n      this._calculateSpacerSize();\n    }\n  }\n  private _orientation: 'horizontal' | 'vertical' = 'vertical';\n\n  /**\n   * Whether rendered items should persist in the DOM after scrolling out of view. By default, items\n   * will be removed.\n   */\n  @Input()\n  get appendOnly(): boolean {\n    return this._appendOnly;\n  }\n  set appendOnly(value: BooleanInput) {\n    this._appendOnly = coerceBooleanProperty(value);\n  }\n  private _appendOnly = false;\n\n  // Note: we don't use the typical EventEmitter here because we need to subscribe to the scroll\n  // strategy lazily (i.e. only if the user is actually listening to the events). We do this because\n  // depending on how the strategy calculates the scrolled index, it may come at a cost to\n  // performance.\n  /** Emits when the index of the first element visible in the viewport changes. */\n  @Output()\n  readonly scrolledIndexChange: Observable<number> = new Observable((observer: Observer<number>) =>\n    this._scrollStrategy.scrolledIndexChange.subscribe(index =>\n      Promise.resolve().then(() => this.ngZone.run(() => observer.next(index))),\n    ),\n  );\n\n  /** The element that wraps the rendered content. */\n  @ViewChild('contentWrapper', {static: true}) _contentWrapper: ElementRef<HTMLElement>;\n\n  /** A stream that emits whenever the rendered range changes. */\n  readonly renderedRangeStream: Observable<ListRange> = this._renderedRangeSubject;\n\n  /**\n   * The total size of all content (in pixels), including content that is not currently rendered.\n   */\n  private _totalContentSize = 0;\n\n  /** A string representing the `style.width` property value to be used for the spacer element. */\n  _totalContentWidth = '';\n\n  /** A string representing the `style.height` property value to be used for the spacer element. */\n  _totalContentHeight = '';\n\n  /**\n   * The CSS transform applied to the rendered subset of items so that they appear within the bounds\n   * of the visible viewport.\n   */\n  private _renderedContentTransform: string;\n\n  /** The currently rendered range of indices. */\n  private _renderedRange: ListRange = {start: 0, end: 0};\n\n  /** The length of the data bound to this viewport (in number of items). */\n  private _dataLength = 0;\n\n  /** The size of the viewport (in pixels). */\n  private _viewportSize = 0;\n\n  /** the currently attached CdkVirtualScrollRepeater. */\n  private _forOf: CdkVirtualScrollRepeater<any> | null;\n\n  /** The last rendered content offset that was set. */\n  private _renderedContentOffset = 0;\n\n  /**\n   * Whether the last rendered content offset was to the end of the content (and therefore needs to\n   * be rewritten as an offset to the start of the content).\n   */\n  private _renderedContentOffsetNeedsRewrite = false;\n\n  /** Whether there is a pending change detection cycle. */\n  private _isChangeDetectionPending = false;\n\n  /** A list of functions to run after the next change detection cycle. */\n  private _runAfterChangeDetection: Function[] = [];\n\n  /** Subscription to changes in the viewport size. */\n  private _viewportChanges = Subscription.EMPTY;\n\n  constructor(\n    public override elementRef: ElementRef<HTMLElement>,\n    private _changeDetectorRef: ChangeDetectorRef,\n    ngZone: NgZone,\n    @Optional()\n    @Inject(VIRTUAL_SCROLL_STRATEGY)\n    private _scrollStrategy: VirtualScrollStrategy,\n    @Optional() dir: Directionality,\n    scrollDispatcher: ScrollDispatcher,\n    viewportRuler: ViewportRuler,\n    @Optional() @Inject(VIRTUAL_SCROLLABLE) public scrollable: CdkVirtualScrollable,\n  ) {\n    super(elementRef, scrollDispatcher, ngZone, dir);\n\n    if (!_scrollStrategy && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('Error: cdk-virtual-scroll-viewport requires the \"itemSize\" property to be set.');\n    }\n\n    this._viewportChanges = viewportRuler.change().subscribe(() => {\n      this.checkViewportSize();\n    });\n\n    if (!this.scrollable) {\n      // No scrollable is provided, so the virtual-scroll-viewport needs to become a scrollable\n      this.elementRef.nativeElement.classList.add('cdk-virtual-scrollable');\n      this.scrollable = this;\n    }\n  }\n\n  override ngOnInit() {\n    // Scrolling depends on the element dimensions which we can't get during SSR.\n    if (!this._platform.isBrowser) {\n      return;\n    }\n\n    if (this.scrollable === this) {\n      super.ngOnInit();\n    }\n    // It's still too early to measure the viewport at this point. Deferring with a promise allows\n    // the Viewport to be rendered with the correct size before we measure. We run this outside the\n    // zone to avoid causing more change detection cycles. We handle the change detection loop\n    // ourselves instead.\n    this.ngZone.runOutsideAngular(() =>\n      Promise.resolve().then(() => {\n        this._measureViewportSize();\n        this._scrollStrategy.attach(this);\n\n        this.scrollable\n          .elementScrolled()\n          .pipe(\n            // Start off with a fake scroll event so we properly detect our initial position.\n            startWith(null),\n            // Collect multiple events into one until the next animation frame. This way if\n            // there are multiple scroll events in the same frame we only need to recheck\n            // our layout once.\n            auditTime(0, SCROLL_SCHEDULER),\n          )\n          .subscribe(() => this._scrollStrategy.onContentScrolled());\n\n        this._markChangeDetectionNeeded();\n      }),\n    );\n  }\n\n  override ngOnDestroy() {\n    this.detach();\n    this._scrollStrategy.detach();\n\n    // Complete all subjects\n    this._renderedRangeSubject.complete();\n    this._detachedSubject.complete();\n    this._viewportChanges.unsubscribe();\n\n    super.ngOnDestroy();\n  }\n\n  /** Attaches a `CdkVirtualScrollRepeater` to this viewport. */\n  attach(forOf: CdkVirtualScrollRepeater<any>) {\n    if (this._forOf && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('CdkVirtualScrollViewport is already attached.');\n    }\n\n    // Subscribe to the data stream of the CdkVirtualForOf to keep track of when the data length\n    // changes. Run outside the zone to avoid triggering change detection, since we're managing the\n    // change detection loop ourselves.\n    this.ngZone.runOutsideAngular(() => {\n      this._forOf = forOf;\n      this._forOf.dataStream.pipe(takeUntil(this._detachedSubject)).subscribe(data => {\n        const newLength = data.length;\n        if (newLength !== this._dataLength) {\n          this._dataLength = newLength;\n          this._scrollStrategy.onDataLengthChanged();\n        }\n        this._doChangeDetection();\n      });\n    });\n  }\n\n  /** Detaches the current `CdkVirtualForOf`. */\n  detach() {\n    this._forOf = null;\n    this._detachedSubject.next();\n  }\n\n  /** Gets the length of the data bound to this viewport (in number of items). */\n  getDataLength(): number {\n    return this._dataLength;\n  }\n\n  /** Gets the size of the viewport (in pixels). */\n  getViewportSize(): number {\n    return this._viewportSize;\n  }\n\n  // TODO(mmalerba): This is technically out of sync with what's really rendered until a render\n  // cycle happens. I'm being careful to only call it after the render cycle is complete and before\n  // setting it to something else, but its error prone and should probably be split into\n  // `pendingRange` and `renderedRange`, the latter reflecting whats actually in the DOM.\n\n  /** Get the current rendered range of items. */\n  getRenderedRange(): ListRange {\n    return this._renderedRange;\n  }\n\n  measureBoundingClientRectWithScrollOffset(from: 'left' | 'top' | 'right' | 'bottom'): number {\n    return this.getElementRef().nativeElement.getBoundingClientRect()[from];\n  }\n\n  /**\n   * Sets the total size of all content (in pixels), including content that is not currently\n   * rendered.\n   */\n  setTotalContentSize(size: number) {\n    if (this._totalContentSize !== size) {\n      this._totalContentSize = size;\n      this._calculateSpacerSize();\n      this._markChangeDetectionNeeded();\n    }\n  }\n\n  /** Sets the currently rendered range of indices. */\n  setRenderedRange(range: ListRange) {\n    if (!rangesEqual(this._renderedRange, range)) {\n      if (this.appendOnly) {\n        range = {start: 0, end: Math.max(this._renderedRange.end, range.end)};\n      }\n      this._renderedRangeSubject.next((this._renderedRange = range));\n      this._markChangeDetectionNeeded(() => this._scrollStrategy.onContentRendered());\n    }\n  }\n\n  /**\n   * Gets the offset from the start of the viewport to the start of the rendered data (in pixels).\n   */\n  getOffsetToRenderedContentStart(): number | null {\n    return this._renderedContentOffsetNeedsRewrite ? null : this._renderedContentOffset;\n  }\n\n  /**\n   * Sets the offset from the start of the viewport to either the start or end of the rendered data\n   * (in pixels).\n   */\n  setRenderedContentOffset(offset: number, to: 'to-start' | 'to-end' = 'to-start') {\n    // In appendOnly, we always start from the top\n    offset = this.appendOnly && to === 'to-start' ? 0 : offset;\n\n    // For a horizontal viewport in a right-to-left language we need to translate along the x-axis\n    // in the negative direction.\n    const isRtl = this.dir && this.dir.value == 'rtl';\n    const isHorizontal = this.orientation == 'horizontal';\n    const axis = isHorizontal ? 'X' : 'Y';\n    const axisDirection = isHorizontal && isRtl ? -1 : 1;\n    let transform = `translate${axis}(${Number(axisDirection * offset)}px)`;\n    this._renderedContentOffset = offset;\n    if (to === 'to-end') {\n      transform += ` translate${axis}(-100%)`;\n      // The viewport should rewrite this as a `to-start` offset on the next render cycle. Otherwise\n      // elements will appear to expand in the wrong direction (e.g. `mat-expansion-panel` would\n      // expand upward).\n      this._renderedContentOffsetNeedsRewrite = true;\n    }\n    if (this._renderedContentTransform != transform) {\n      // We know this value is safe because we parse `offset` with `Number()` before passing it\n      // into the string.\n      this._renderedContentTransform = transform;\n      this._markChangeDetectionNeeded(() => {\n        if (this._renderedContentOffsetNeedsRewrite) {\n          this._renderedContentOffset -= this.measureRenderedContentSize();\n          this._renderedContentOffsetNeedsRewrite = false;\n          this.setRenderedContentOffset(this._renderedContentOffset);\n        } else {\n          this._scrollStrategy.onRenderedOffsetChanged();\n        }\n      });\n    }\n  }\n\n  /**\n   * Scrolls to the given offset from the start of the viewport. Please note that this is not always\n   * the same as setting `scrollTop` or `scrollLeft`. In a horizontal viewport with right-to-left\n   * direction, this would be the equivalent of setting a fictional `scrollRight` property.\n   * @param offset The offset to scroll to.\n   * @param behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.\n   */\n  scrollToOffset(offset: number, behavior: ScrollBehavior = 'auto') {\n    const options: ExtendedScrollToOptions = {behavior};\n    if (this.orientation === 'horizontal') {\n      options.start = offset;\n    } else {\n      options.top = offset;\n    }\n    this.scrollable.scrollTo(options);\n  }\n\n  /**\n   * Scrolls to the offset for the given index.\n   * @param index The index of the element to scroll to.\n   * @param behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.\n   */\n  scrollToIndex(index: number, behavior: ScrollBehavior = 'auto') {\n    this._scrollStrategy.scrollToIndex(index, behavior);\n  }\n\n  /**\n   * Gets the current scroll offset from the start of the scrollable (in pixels).\n   * @param from The edge to measure the offset from. Defaults to 'top' in vertical mode and 'start'\n   *     in horizontal mode.\n   */\n  override measureScrollOffset(\n    from?: 'top' | 'left' | 'right' | 'bottom' | 'start' | 'end',\n  ): number {\n    // This is to break the call cycle\n    let measureScrollOffset: InstanceType<typeof CdkVirtualScrollable>['measureScrollOffset'];\n    if (this.scrollable == this) {\n      measureScrollOffset = (_from: NonNullable<typeof from>) => super.measureScrollOffset(_from);\n    } else {\n      measureScrollOffset = (_from: NonNullable<typeof from>) =>\n        this.scrollable.measureScrollOffset(_from);\n    }\n\n    return Math.max(\n      0,\n      measureScrollOffset(from ?? (this.orientation === 'horizontal' ? 'start' : 'top')) -\n        this.measureViewportOffset(),\n    );\n  }\n\n  /**\n   * Measures the offset of the viewport from the scrolling container\n   * @param from The edge to measure from.\n   */\n  measureViewportOffset(from?: 'top' | 'left' | 'right' | 'bottom' | 'start' | 'end') {\n    let fromRect: 'left' | 'top' | 'right' | 'bottom';\n    const LEFT = 'left';\n    const RIGHT = 'right';\n    const isRtl = this.dir?.value == 'rtl';\n    if (from == 'start') {\n      fromRect = isRtl ? RIGHT : LEFT;\n    } else if (from == 'end') {\n      fromRect = isRtl ? LEFT : RIGHT;\n    } else if (from) {\n      fromRect = from;\n    } else {\n      fromRect = this.orientation === 'horizontal' ? 'left' : 'top';\n    }\n\n    const scrollerClientRect = this.scrollable.measureBoundingClientRectWithScrollOffset(fromRect);\n    const viewportClientRect = this.elementRef.nativeElement.getBoundingClientRect()[fromRect];\n\n    return viewportClientRect - scrollerClientRect;\n  }\n\n  /** Measure the combined size of all of the rendered items. */\n  measureRenderedContentSize(): number {\n    const contentEl = this._contentWrapper.nativeElement;\n    return this.orientation === 'horizontal' ? contentEl.offsetWidth : contentEl.offsetHeight;\n  }\n\n  /**\n   * Measure the total combined size of the given range. Throws if the range includes items that are\n   * not rendered.\n   */\n  measureRangeSize(range: ListRange): number {\n    if (!this._forOf) {\n      return 0;\n    }\n    return this._forOf.measureRangeSize(range, this.orientation);\n  }\n\n  /** Update the viewport dimensions and re-render. */\n  checkViewportSize() {\n    // TODO: Cleanup later when add logic for handling content resize\n    this._measureViewportSize();\n    this._scrollStrategy.onDataLengthChanged();\n  }\n\n  /** Measure the viewport size. */\n  private _measureViewportSize() {\n    this._viewportSize = this.scrollable.measureViewportSize(this.orientation);\n  }\n\n  /** Queue up change detection to run. */\n  private _markChangeDetectionNeeded(runAfter?: Function) {\n    if (runAfter) {\n      this._runAfterChangeDetection.push(runAfter);\n    }\n\n    // Use a Promise to batch together calls to `_doChangeDetection`. This way if we set a bunch of\n    // properties sequentially we only have to run `_doChangeDetection` once at the end.\n    if (!this._isChangeDetectionPending) {\n      this._isChangeDetectionPending = true;\n      this.ngZone.runOutsideAngular(() =>\n        Promise.resolve().then(() => {\n          this._doChangeDetection();\n        }),\n      );\n    }\n  }\n\n  /** Run change detection. */\n  private _doChangeDetection() {\n    this._isChangeDetectionPending = false;\n\n    // Apply the content transform. The transform can't be set via an Angular binding because\n    // bypassSecurityTrustStyle is banned in Google. However the value is safe, it's composed of\n    // string literals, a variable that can only be 'X' or 'Y', and user input that is run through\n    // the `Number` function first to coerce it to a numeric value.\n    this._contentWrapper.nativeElement.style.transform = this._renderedContentTransform;\n    // Apply changes to Angular bindings. Note: We must call `markForCheck` to run change detection\n    // from the root, since the repeated items are content projected in. Calling `detectChanges`\n    // instead does not properly check the projected content.\n    this.ngZone.run(() => this._changeDetectorRef.markForCheck());\n\n    const runAfterChangeDetection = this._runAfterChangeDetection;\n    this._runAfterChangeDetection = [];\n    for (const fn of runAfterChangeDetection) {\n      fn();\n    }\n  }\n\n  /** Calculates the `style.width` and `style.height` for the spacer element. */\n  private _calculateSpacerSize() {\n    this._totalContentHeight =\n      this.orientation === 'horizontal' ? '' : `${this._totalContentSize}px`;\n    this._totalContentWidth =\n      this.orientation === 'horizontal' ? `${this._totalContentSize}px` : '';\n  }\n}\n","<!--\n  Wrap the rendered content in an element that will be used to offset it based on the scroll\n  position.\n-->\n<div #contentWrapper class=\"cdk-virtual-scroll-content-wrapper\">\n  <ng-content></ng-content>\n</div>\n<!--\n  Spacer used to force the scrolling container to the correct size for the *total* number of items\n  so that the scrollbar captures the size of the entire data set.\n-->\n<div class=\"cdk-virtual-scroll-spacer\"\n     [style.width]=\"_totalContentWidth\" [style.height]=\"_totalContentHeight\"></div>\n"]} |
\ | No newline at end of file |