You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
149 lines
21 KiB
149 lines
21 KiB
/**
|
|
* @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 { Platform } from '@angular/cdk/platform';
|
|
import { Injectable, NgZone, Optional, Inject } from '@angular/core';
|
|
import { Subject } from 'rxjs';
|
|
import { auditTime } from 'rxjs/operators';
|
|
import { DOCUMENT } from '@angular/common';
|
|
import * as i0 from "@angular/core";
|
|
import * as i1 from "@angular/cdk/platform";
|
|
import * as i2 from "@angular/common";
|
|
/** Time in ms to throttle the resize events by default. */
|
|
import * as ɵngcc0 from '@angular/core';
|
|
import * as ɵngcc1 from '@angular/cdk/platform';
|
|
export const DEFAULT_RESIZE_TIME = 20;
|
|
/**
|
|
* Simple utility for getting the bounds of the browser viewport.
|
|
* @docs-private
|
|
*/
|
|
export class ViewportRuler {
|
|
constructor(_platform, ngZone, document) {
|
|
this._platform = _platform;
|
|
/** Stream of viewport change events. */
|
|
this._change = new Subject();
|
|
/** Event listener that will be used to handle the viewport change events. */
|
|
this._changeListener = (event) => {
|
|
this._change.next(event);
|
|
};
|
|
this._document = document;
|
|
ngZone.runOutsideAngular(() => {
|
|
if (_platform.isBrowser) {
|
|
const window = this._getWindow();
|
|
// Note that bind the events ourselves, rather than going through something like RxJS's
|
|
// `fromEvent` so that we can ensure that they're bound outside of the NgZone.
|
|
window.addEventListener('resize', this._changeListener);
|
|
window.addEventListener('orientationchange', this._changeListener);
|
|
}
|
|
// Clear the cached position so that the viewport is re-measured next time it is required.
|
|
// We don't need to keep track of the subscription, because it is completed on destroy.
|
|
this.change().subscribe(() => this._viewportSize = null);
|
|
});
|
|
}
|
|
ngOnDestroy() {
|
|
if (this._platform.isBrowser) {
|
|
const window = this._getWindow();
|
|
window.removeEventListener('resize', this._changeListener);
|
|
window.removeEventListener('orientationchange', this._changeListener);
|
|
}
|
|
this._change.complete();
|
|
}
|
|
/** Returns the viewport's width and height. */
|
|
getViewportSize() {
|
|
if (!this._viewportSize) {
|
|
this._updateViewportSize();
|
|
}
|
|
const output = { width: this._viewportSize.width, height: this._viewportSize.height };
|
|
// If we're not on a browser, don't cache the size since it'll be mocked out anyway.
|
|
if (!this._platform.isBrowser) {
|
|
this._viewportSize = null;
|
|
}
|
|
return output;
|
|
}
|
|
/** Gets a ClientRect for the viewport's bounds. */
|
|
getViewportRect() {
|
|
// Use the document element's bounding rect rather than the window scroll properties
|
|
// (e.g. pageYOffset, scrollY) due to in issue in Chrome and IE where window scroll
|
|
// properties and client coordinates (boundingClientRect, clientX/Y, etc.) are in different
|
|
// conceptual viewports. Under most circumstances these viewports are equivalent, but they
|
|
// can disagree when the page is pinch-zoomed (on devices that support touch).
|
|
// See https://bugs.chromium.org/p/chromium/issues/detail?id=489206#c4
|
|
// We use the documentElement instead of the body because, by default (without a css reset)
|
|
// browsers typically give the document body an 8px margin, which is not included in
|
|
// getBoundingClientRect().
|
|
const scrollPosition = this.getViewportScrollPosition();
|
|
const { width, height } = this.getViewportSize();
|
|
return {
|
|
top: scrollPosition.top,
|
|
left: scrollPosition.left,
|
|
bottom: scrollPosition.top + height,
|
|
right: scrollPosition.left + width,
|
|
height,
|
|
width,
|
|
};
|
|
}
|
|
/** Gets the (top, left) scroll position of the viewport. */
|
|
getViewportScrollPosition() {
|
|
// While we can get a reference to the fake document
|
|
// during SSR, it doesn't have getBoundingClientRect.
|
|
if (!this._platform.isBrowser) {
|
|
return { top: 0, left: 0 };
|
|
}
|
|
// The top-left-corner of the viewport is determined by the scroll position of the document
|
|
// body, normally just (scrollLeft, scrollTop). However, Chrome and Firefox disagree about
|
|
// whether `document.body` or `document.documentElement` is the scrolled element, so reading
|
|
// `scrollTop` and `scrollLeft` is inconsistent. However, using the bounding rect of
|
|
// `document.documentElement` works consistently, where the `top` and `left` values will
|
|
// equal negative the scroll position.
|
|
const document = this._document;
|
|
const window = this._getWindow();
|
|
const documentElement = document.documentElement;
|
|
const documentRect = documentElement.getBoundingClientRect();
|
|
const top = -documentRect.top || document.body.scrollTop || window.scrollY ||
|
|
documentElement.scrollTop || 0;
|
|
const left = -documentRect.left || document.body.scrollLeft || window.scrollX ||
|
|
documentElement.scrollLeft || 0;
|
|
return { top, left };
|
|
}
|
|
/**
|
|
* Returns a stream that emits whenever the size of the viewport changes.
|
|
* This stream emits outside of the Angular zone.
|
|
* @param throttleTime Time in milliseconds to throttle the stream.
|
|
*/
|
|
change(throttleTime = DEFAULT_RESIZE_TIME) {
|
|
return throttleTime > 0 ? this._change.pipe(auditTime(throttleTime)) : this._change;
|
|
}
|
|
/** Use defaultView of injected document if available or fallback to global window reference */
|
|
_getWindow() {
|
|
return this._document.defaultView || window;
|
|
}
|
|
/** Updates the cached viewport size. */
|
|
_updateViewportSize() {
|
|
const window = this._getWindow();
|
|
this._viewportSize = this._platform.isBrowser ?
|
|
{ width: window.innerWidth, height: window.innerHeight } :
|
|
{ width: 0, height: 0 };
|
|
}
|
|
}
|
|
ViewportRuler.ɵfac = function ViewportRuler_Factory(t) { return new (t || ViewportRuler)(ɵngcc0.ɵɵinject(ɵngcc1.Platform), ɵngcc0.ɵɵinject(ɵngcc0.NgZone), ɵngcc0.ɵɵinject(DOCUMENT, 8)); };
|
|
ViewportRuler.ɵprov = i0.ɵɵdefineInjectable({ factory: function ViewportRuler_Factory() { return new ViewportRuler(i0.ɵɵinject(i1.Platform), i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i2.DOCUMENT, 8)); }, token: ViewportRuler, providedIn: "root" });
|
|
ViewportRuler.ctorParameters = () => [
|
|
{ type: Platform },
|
|
{ type: NgZone },
|
|
{ type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] }
|
|
];
|
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(ViewportRuler, [{
|
|
type: Injectable,
|
|
args: [{ providedIn: 'root' }]
|
|
}], function () { return [{ type: ɵngcc1.Platform }, { type: ɵngcc0.NgZone }, { type: undefined, decorators: [{
|
|
type: Optional
|
|
}, {
|
|
type: Inject,
|
|
args: [DOCUMENT]
|
|
}] }]; }, null); })();
|
|
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"viewport-ruler.js","sources":["../../../../../../src/cdk/scrolling/viewport-ruler.ts"],"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 {Platform} from '@angular/cdk/platform';\nimport {Injectable, NgZone, OnDestroy, Optional, Inject} from '@angular/core';\nimport {Observable, Subject} from 'rxjs';\nimport {auditTime} from 'rxjs/operators';\nimport {DOCUMENT} from '@angular/common';\n\n/** Time in ms to throttle the resize events by default. */\nexport const DEFAULT_RESIZE_TIME = 20;\n\n/** Object that holds the scroll position of the viewport in each direction. */\nexport interface ViewportScrollPosition {\n  top: number;\n  left: number;\n}\n\n/**\n * Simple utility for getting the bounds of the browser viewport.\n * @docs-private\n */\n@Injectable({providedIn: 'root'})\nexport class ViewportRuler implements OnDestroy {\n  /** Cached viewport dimensions. */\n  private _viewportSize: {width: number; height: number} | null;\n\n  /** Stream of viewport change events. */\n  private readonly _change = new Subject<Event>();\n\n  /** Event listener that will be used to handle the viewport change events. */\n  private _changeListener = (event: Event) => {\n    this._change.next(event);\n  }\n\n  /** Used to reference correct document/window */\n  protected _document: Document;\n\n  constructor(private _platform: Platform,\n              ngZone: NgZone,\n              @Optional() @Inject(DOCUMENT) document: any) {\n    this._document = document;\n\n    ngZone.runOutsideAngular(() => {\n      if (_platform.isBrowser) {\n        const window = this._getWindow();\n\n        // Note that bind the events ourselves, rather than going through something like RxJS's\n        // `fromEvent` so that we can ensure that they're bound outside of the NgZone.\n        window.addEventListener('resize', this._changeListener);\n        window.addEventListener('orientationchange', this._changeListener);\n      }\n\n      // Clear the cached position so that the viewport is re-measured next time it is required.\n      // We don't need to keep track of the subscription, because it is completed on destroy.\n      this.change().subscribe(() => this._viewportSize = null);\n    });\n  }\n\n  ngOnDestroy() {\n    if (this._platform.isBrowser) {\n      const window = this._getWindow();\n      window.removeEventListener('resize', this._changeListener);\n      window.removeEventListener('orientationchange', this._changeListener);\n    }\n\n    this._change.complete();\n  }\n\n  /** Returns the viewport's width and height. */\n  getViewportSize(): Readonly<{width: number, height: number}> {\n    if (!this._viewportSize) {\n      this._updateViewportSize();\n    }\n\n    const output = {width: this._viewportSize!.width, height: this._viewportSize!.height};\n\n    // If we're not on a browser, don't cache the size since it'll be mocked out anyway.\n    if (!this._platform.isBrowser) {\n      this._viewportSize = null!;\n    }\n\n    return output;\n  }\n\n  /** Gets a ClientRect for the viewport's bounds. */\n  getViewportRect(): ClientRect {\n    // Use the document element's bounding rect rather than the window scroll properties\n    // (e.g. pageYOffset, scrollY) due to in issue in Chrome and IE where window scroll\n    // properties and client coordinates (boundingClientRect, clientX/Y, etc.) are in different\n    // conceptual viewports. Under most circumstances these viewports are equivalent, but they\n    // can disagree when the page is pinch-zoomed (on devices that support touch).\n    // See https://bugs.chromium.org/p/chromium/issues/detail?id=489206#c4\n    // We use the documentElement instead of the body because, by default (without a css reset)\n    // browsers typically give the document body an 8px margin, which is not included in\n    // getBoundingClientRect().\n    const scrollPosition = this.getViewportScrollPosition();\n    const {width, height} = this.getViewportSize();\n\n    return {\n      top: scrollPosition.top,\n      left: scrollPosition.left,\n      bottom: scrollPosition.top + height,\n      right: scrollPosition.left + width,\n      height,\n      width,\n    };\n  }\n\n  /** Gets the (top, left) scroll position of the viewport. */\n  getViewportScrollPosition(): ViewportScrollPosition {\n    // While we can get a reference to the fake document\n    // during SSR, it doesn't have getBoundingClientRect.\n    if (!this._platform.isBrowser) {\n      return {top: 0, left: 0};\n    }\n\n    // The top-left-corner of the viewport is determined by the scroll position of the document\n    // body, normally just (scrollLeft, scrollTop). However, Chrome and Firefox disagree about\n    // whether `document.body` or `document.documentElement` is the scrolled element, so reading\n    // `scrollTop` and `scrollLeft` is inconsistent. However, using the bounding rect of\n    // `document.documentElement` works consistently, where the `top` and `left` values will\n    // equal negative the scroll position.\n    const document = this._document;\n    const window = this._getWindow();\n    const documentElement = document.documentElement!;\n    const documentRect = documentElement.getBoundingClientRect();\n\n    const top = -documentRect.top || document.body.scrollTop || window.scrollY ||\n                 documentElement.scrollTop || 0;\n\n    const left = -documentRect.left || document.body.scrollLeft || window.scrollX ||\n                  documentElement.scrollLeft || 0;\n\n    return {top, left};\n  }\n\n  /**\n   * Returns a stream that emits whenever the size of the viewport changes.\n   * This stream emits outside of the Angular zone.\n   * @param throttleTime Time in milliseconds to throttle the stream.\n   */\n  change(throttleTime: number = DEFAULT_RESIZE_TIME): Observable<Event> {\n    return throttleTime > 0 ? this._change.pipe(auditTime(throttleTime)) : this._change;\n  }\n\n  /** Use defaultView of injected document if available or fallback to global window reference */\n  private _getWindow(): Window {\n    return this._document.defaultView || window;\n  }\n\n  /** Updates the cached viewport size. */\n  private _updateViewportSize() {\n    const window = this._getWindow();\n    this._viewportSize = this._platform.isBrowser ?\n        {width: window.innerWidth, height: window.innerHeight} :\n        {width: 0, height: 0};\n  }\n}\n"]}
|