UNPKG

@ngez/core

Version:

A collection of minimalistic, easy-to-use and fully customizable Angular components, directives and services

231 lines 21.3 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import * as tslib_1 from "tslib"; import { Directive, ElementRef, Inject, PLATFORM_ID, Optional, Input, Output, EventEmitter } from "@angular/core"; import { WINDOW } from "../window/window.service"; import { DOCUMENT, isPlatformBrowser } from "@angular/common"; import { defaultConfig, defaultOffsetConfig } from "./models"; import { fromEvent, merge, of } from "rxjs"; export class NgEzInViewportDirective { /** * @param {?} element * @param {?} platformId * @param {?} window * @param {?} document */ constructor(element, platformId, window, document) { this.element = element; this.platformId = platformId; this.window = window; this.document = document; this.inViewportChange = new EventEmitter(); this.latest = { top: false, bottom: false, left: false, right: false, all: false, any: false }; } /** * @param {?} config * @return {?} */ set config(config) { this._config = config; } /** * @return {?} */ ngOnInit() { if (!isPlatformBrowser(this.platformId)) return; this.container = this.getClosestScrollableParent(this.element.nativeElement); //Check on scroll, window resize and oninit this.subscription = merge(fromEvent(this.isDocumentTheScrollableContainer() ? this.document : this.container, 'scroll'), fromEvent(this.window, 'resize'), of(null)) .subscribe(() => this.check()); } /** * @return {?} */ ngOnDestroy() { if (this.subscription) this.subscription.unsubscribe(); } /** * @return {?} */ get config() { const _a = this._config || {}, { offset = {} } = _a, config = tslib_1.__rest(_a, ["offset"]); return Object.assign({}, defaultConfig, config, { offset: Object.assign({}, defaultOffsetConfig, offset) }); } /** * @return {?} */ check() { /** @type {?} */ const previous = this.latest; /** @type {?} */ const current = this.calculateVisibility(); /** @type {?} */ const hasChanged = previous.top != current.top || previous.bottom != current.bottom || previous.left != current.left || previous.right != current.right; if (hasChanged) this.inViewportChange.emit(current); this.latest = current; return current; } /** * @private * @return {?} */ calculateVisibility() { /** @type {?} */ const elementRect = this.element.nativeElement.getBoundingClientRect(); /** @type {?} */ const height = elementRect.height; /** @type {?} */ const width = elementRect.width; /** @type {?} */ const containerRect = this.container.getBoundingClientRect(); /** @type {?} */ const top = elementRect.top - (this.isDocumentTheScrollableContainer() ? 0 : containerRect.top); /** @type {?} */ const bottom = top + height; /** @type {?} */ const left = elementRect.left - (this.isDocumentTheScrollableContainer() ? 0 : containerRect.left); /** @type {?} */ const right = left + width; /** @type {?} */ const containerHeight = this.container.clientHeight || this.window.innerHeight; /** @type {?} */ const containerWidth = this.container.clientWidth || this.window.innerWidth; /** @type {?} */ const topInView = top + this.config.offset.top >= 0 && top <= (containerHeight + this.config.offset.top); /** @type {?} */ const bottomInView = bottom + this.config.offset.bottom >= 0 && bottom <= (containerHeight + this.config.offset.bottom); /** @type {?} */ const leftInView = left + this.config.offset.left >= 0 && left <= (containerWidth + this.config.offset.left); /** @type {?} */ const rightInView = right + this.config.offset.right >= 0 && right <= (containerWidth + this.config.offset.right); return { top: topInView, bottom: bottomInView, left: leftInView && (topInView || bottomInView), right: rightInView && (topInView || bottomInView), any: (topInView || bottomInView) && (leftInView || rightInView), all: topInView && bottomInView && leftInView && rightInView }; } /** * @private * @param {?} node * @return {?} */ getClosestScrollableParent(node) { return !node || node === this.document.body ? this.document.documentElement : this.isScrollable(node) ? node : this.getClosestScrollableParent(node.parentNode); } /** * @private * @param {?} node * @return {?} */ isScrollable(node) { /** @type {?} */ const regex = /(auto|scroll)/; return regex.test(this.getComputedStyle(node, 'overflow') + this.getComputedStyle(node, 'overflow-y') + this.getComputedStyle(node, 'overflow-x')); } /** * @private * @param {?} node * @param {?} style * @return {?} */ getComputedStyle(node, style) { return this.window.getComputedStyle(node).getPropertyValue(style); } /** * @private * @return {?} */ isDocumentTheScrollableContainer() { return this.document.documentElement === this.container; } } NgEzInViewportDirective.decorators = [ { type: Directive, args: [{ selector: '[ngezInViewport]', exportAs: 'ngezInViewport' },] } ]; /** @nocollapse */ NgEzInViewportDirective.ctorParameters = () => [ { type: ElementRef }, { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }, { type: undefined, decorators: [{ type: Inject, args: [WINDOW,] }] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] } ]; NgEzInViewportDirective.propDecorators = { config: [{ type: Input }], inViewportChange: [{ type: Output }] }; if (false) { /** @type {?} */ NgEzInViewportDirective.prototype.inViewportChange; /** * @type {?} * @private */ NgEzInViewportDirective.prototype._config; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.container; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.subscription; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.latest; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.element; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.platformId; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.window; /** * @type {?} * @private */ NgEzInViewportDirective.prototype.document; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"in-viewport.directive.js","sourceRoot":"ng://@ngez/core/","sources":["in-viewport/in-viewport.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { \r\n    Directive, \r\n    ElementRef, \r\n    Inject, \r\n    PLATFORM_ID, \r\n    Optional, \r\n    OnInit, \r\n    Input, \r\n    TemplateRef, \r\n    OnDestroy, \r\n    Output,\r\n    EventEmitter } from \"@angular/core\";\r\nimport { WINDOW } from \"../window/window.service\";\r\nimport { DOCUMENT, isPlatformBrowser } from \"@angular/common\";\r\nimport { NgEzInViewportConfig, defaultConfig, defaultOffsetConfig, NgEzInViewportEvent } from \"./models\";\r\nimport { fromEvent, merge, Subscription, of } from \"rxjs\";\r\n\r\n@Directive({\r\n    selector: '[ngezInViewport]',\r\n    exportAs: 'ngezInViewport'\r\n})\r\nexport class NgEzInViewportDirective implements OnInit, OnDestroy{\r\n\r\n    @Input() set config(config: NgEzInViewportConfig){\r\n        this._config = config;\r\n    }\r\n\r\n    @Output() inViewportChange = new EventEmitter<NgEzInViewportEvent>();\r\n\r\n    private _config: NgEzInViewportConfig;\r\n\r\n    private container: Element;\r\n\r\n    private subscription: Subscription;\r\n\r\n    private latest: NgEzInViewportEvent = {\r\n        top: false,\r\n        bottom: false,\r\n        left: false,\r\n        right: false,\r\n        all: false,\r\n        any: false\r\n    };\r\n\r\n    constructor(\r\n        private element: ElementRef,\r\n        @Inject(PLATFORM_ID) private platformId: Object,\r\n        @Inject(WINDOW) private window: any,\r\n        @Optional() @Inject(DOCUMENT) private document: any){}\r\n\r\n    ngOnInit() {\r\n        if(!isPlatformBrowser(this.platformId)) return;\r\n\r\n        this.container = this.getClosestScrollableParent(this.element.nativeElement);\r\n        \r\n        //Check on scroll, window resize and oninit\r\n        this.subscription = merge(\r\n            fromEvent(this.isDocumentTheScrollableContainer() ? this.document : this.container, 'scroll'), \r\n            fromEvent(this.window, 'resize'),\r\n            of(null))\r\n                .subscribe(() => this.check());\r\n    }\r\n\r\n    ngOnDestroy() {\r\n        if(this.subscription)\r\n            this.subscription.unsubscribe();\r\n    }\r\n\r\n\r\n    get config(): NgEzInViewportConfig {\r\n        const { offset = {}, ...config } = this._config || {};\r\n\r\n        return { \r\n            ...defaultConfig, \r\n            ...config, \r\n            offset: {\r\n                ...defaultOffsetConfig,\r\n                ...offset\r\n            }\r\n        };\r\n    }\r\n\r\n    check(): NgEzInViewportEvent{\r\n        const previous = this.latest;\r\n        const current = this.calculateVisibility();\r\n\r\n        const hasChanged = previous.top != current.top\r\n            || previous.bottom != current.bottom\r\n            || previous.left != current.left\r\n            || previous.right != current.right;\r\n\r\n        if(hasChanged)\r\n            this.inViewportChange.emit(current);\r\n\r\n        this.latest = current;\r\n        return current;\r\n    }\r\n\r\n    private calculateVisibility(): NgEzInViewportEvent{\r\n        const elementRect = this.element.nativeElement.getBoundingClientRect();\r\n        \r\n        const height = elementRect.height;\r\n        const width =  elementRect.width;\r\n\r\n        const containerRect = this.container.getBoundingClientRect();\r\n\r\n        const top = elementRect.top - (this.isDocumentTheScrollableContainer()\r\n            ? 0\r\n            : containerRect.top);\r\n\r\n        const bottom = top + height;\r\n\r\n        const left = elementRect.left - (this.isDocumentTheScrollableContainer()\r\n            ? 0\r\n            : containerRect.left);\r\n\r\n        const right = left + width;\r\n\r\n        const containerHeight = this.container.clientHeight || this.window.innerHeight;\r\n        const containerWidth = this.container.clientWidth || this.window.innerWidth;\r\n\r\n        const topInView = top + this.config.offset.top >= 0 && top <= (containerHeight + this.config.offset.top);\r\n        const bottomInView = bottom + this.config.offset.bottom >= 0 && bottom <= (containerHeight + this.config.offset.bottom);\r\n        const leftInView = left + this.config.offset.left >= 0 && left <= (containerWidth + this.config.offset.left);\r\n        const rightInView = right + this.config.offset.right >= 0 && right <= (containerWidth + this.config.offset.right);\r\n\r\n        return {\r\n            top: topInView,\r\n            bottom: bottomInView,\r\n            left: leftInView && (topInView || bottomInView),\r\n            right: rightInView && (topInView || bottomInView),\r\n            any: (topInView || bottomInView) && (leftInView || rightInView),\r\n            all: topInView && bottomInView && leftInView && rightInView\r\n        };\r\n    }\r\n\r\n    private getClosestScrollableParent(node: Node) {\r\n        return !node || node === this.document.body\r\n            ? this.document.documentElement\r\n            : this.isScrollable(node)\r\n                ? node \r\n                : this.getClosestScrollableParent(node.parentNode);\r\n    }\r\n\r\n    private isScrollable(node: Node){\r\n        const regex = /(auto|scroll)/;\r\n        return regex.test(\r\n            this.getComputedStyle(node, 'overflow') +\r\n            this.getComputedStyle(node, 'overflow-y') +\r\n            this.getComputedStyle(node, 'overflow-x'));\r\n    }\r\n\r\n    private getComputedStyle(node: Node, style: string){\r\n        return this.window.getComputedStyle(node).getPropertyValue(style);\r\n    }\r\n\r\n    private isDocumentTheScrollableContainer(){\r\n        return this.document.documentElement === this.container;\r\n    }\r\n}"]}