UNPKG

gentics-ui-core

Version:

This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.

145 lines 19.8 kB
import { Component, ComponentFactoryResolver, HostListener, Renderer2, ViewChild, ViewContainerRef } from '@angular/core'; import { Subject, Subscription } from 'rxjs'; import { debounceTime } from 'rxjs/operators'; import { UserAgentRef } from './user-agent-ref'; import * as i0 from "@angular/core"; import * as i1 from "./user-agent-ref"; const defaultOptions = { modalBodyClass: 'modal-content', padding: true, width: null, closeOnEscape: true, closeOnOverlayClick: true }; /** * This is an internal component which is responsible for creating the modal dialog window and overlay. */ export class DynamicModalWrapper { constructor(componentFactoryResolver, userAgent, renderer) { this.componentFactoryResolver = componentFactoryResolver; this.userAgent = userAgent; this.renderer = renderer; this.modalElementsHeight = 0; this.visible = false; this.options = defaultOptions; this.subscriptions = new Subscription(); this.modalHeightEvents$ = new Subject(); } ngOnInit() { this.isIE11 = this.userAgent.isIE11; if (this.isIE11) { this.subscriptions.add(this.modalHeightEvents$.pipe(debounceTime(100)).subscribe(() => { this.ie11FixContentHeight(); })); } } ngOnDestroy() { this.subscriptions.unsubscribe(); clearTimeout(this.openTimer); if (this.cmpRef && this.cmpRef.destroy) { this.cmpRef.destroy(); } } setOptions(options) { this.options = Object.assign({}, defaultOptions, options); } /** * Inject the component which will appear within the modal. */ injectContent(component) { let factory = this.componentFactoryResolver.resolveComponentFactory(component); this.cmpRef = this.portal.createComponent(factory); return this.cmpRef; } /** * Display the modal */ open() { clearTimeout(this.openTimer); this.openTimer = setTimeout(() => this.visible = true, 50); } /** * Programatically force the modal to close and resolve with the value passed. */ forceClose(val) { this.cmpRef.instance.closeFn(val); } /** * Close the modal and call the cancelFn of the embedded component. */ cancel() { clearTimeout(this.openTimer); this.visible = false; this.cmpRef.instance.cancelFn(); this.cmpRef.destroy(); } overlayClick() { if (this.options.closeOnOverlayClick) { this.cancel(); } } keyHandler(e) { if (e.which === 27 && this.options.closeOnEscape) { this.cancel(); } } /** * IE11 Related fixes */ /** * Listen for browsers size changes, to notify IE for modal height change */ onResize() { if (this.isIE11) { this.modalHeightEvents$.next(); } } /** * Listen for content changes, to notify IE for modal height change */ ngAfterViewChecked() { if (this.isIE11 && this.cmpRef) { // Trigger modalHeight event on view checks let modalElements = Array.from(this.cmpRef.location.nativeElement.children); let currentModalElementsHeight = modalElements .filter(element => element.className != this.options.modalBodyClass) .map(element => { let styles = window.getComputedStyle(element); let margin = parseFloat(styles['marginTop']) + parseFloat(styles['marginBottom']); return element.offsetHeight + margin; }) .reduce((heights, height) => heights + height); if (this.modalElementsHeight !== currentModalElementsHeight) { this.modalElementsHeight = currentModalElementsHeight; this.modalHeightEvents$.next(); } } } /** * Fixes modal body height for IE11 */ ie11FixContentHeight() { if (this.isIE11 && this.cmpRef) { let injectedElement = this.cmpRef.location.nativeElement; let modalBodyElement = injectedElement.getElementsByClassName(this.options.modalBodyClass)[0]; this.renderer.setStyle(modalBodyElement, 'max-height', `calc(70vh - ${this.modalElementsHeight}px)`); } } } /** @nocollapse */ DynamicModalWrapper.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: DynamicModalWrapper, deps: [{ token: i0.ComponentFactoryResolver }, { token: i1.UserAgentRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ DynamicModalWrapper.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: DynamicModalWrapper, selector: "gtx-dynamic-modal", host: { listeners: { "document:keydown": "keyHandler($event)", "window:resize": "onResize()" } }, viewQueries: [{ propertyName: "portal", first: true, predicate: ["portal"], descendants: true, read: ViewContainerRef, static: true }], ngImport: i0, template: "<div class=\"gtx-modal-overlay\"\n [class.visible]=\"visible\"\n (click)=\"overlayClick()\"></div>\n<div class=\"gtx-modal-dialog\"\n [class.isIE11]=\"isIE11\"\n [class.visible]=\"visible\"\n [style.width]=\"options.width\"\n [class.nopad]=\"options.padding === false\">\n <ng-template #portal></ng-template>\n</div>\n" }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: DynamicModalWrapper, decorators: [{ type: Component, args: [{ selector: 'gtx-dynamic-modal', template: "<div class=\"gtx-modal-overlay\"\n [class.visible]=\"visible\"\n (click)=\"overlayClick()\"></div>\n<div class=\"gtx-modal-dialog\"\n [class.isIE11]=\"isIE11\"\n [class.visible]=\"visible\"\n [style.width]=\"options.width\"\n [class.nopad]=\"options.padding === false\">\n <ng-template #portal></ng-template>\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.ComponentFactoryResolver }, { type: i1.UserAgentRef }, { type: i0.Renderer2 }]; }, propDecorators: { portal: [{ type: ViewChild, args: ['portal', { read: ViewContainerRef, static: true }] }], keyHandler: [{ type: HostListener, args: ['document:keydown', ['$event']] }], onResize: [{ type: HostListener, args: ['window:resize'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-modal-wrapper.component.js","sourceRoot":"","sources":["../../../../../src/components/modal/dynamic-modal-wrapper.component.ts","../../../../../src/components/modal/dynamic-modal-wrapper.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewChecked,\n    Component,\n    ComponentFactoryResolver,\n    ComponentRef,\n    HostListener,\n    OnDestroy,\n    OnInit,\n    Renderer2,\n    Type,\n    ViewChild,\n    ViewContainerRef\n} from '@angular/core';\nimport {Subject, Subscription} from 'rxjs';\nimport {debounceTime} from 'rxjs/operators';\n\nimport {IModalDialog, IModalOptions} from './modal-interfaces';\nimport {UserAgentRef} from './user-agent-ref';\n\nconst defaultOptions: IModalOptions = {\n    modalBodyClass: 'modal-content',\n    padding: true,\n    width: null,\n    closeOnEscape: true,\n    closeOnOverlayClick: true\n};\n\n/**\n * This is an internal component which is responsible for creating the modal dialog window and overlay.\n */\n@Component({\n    selector: 'gtx-dynamic-modal',\n    templateUrl: './dynamic-modal-wrapper.tpl.html'\n})\nexport class DynamicModalWrapper implements OnInit, OnDestroy, AfterViewChecked {\n\n    @ViewChild('portal', { read: ViewContainerRef, static: true }) portal: ViewContainerRef;\n\n    isIE11: boolean;\n    dismissFn: Function;\n\n    modalElementsHeight: number = 0;\n    visible: boolean = false;\n    options: IModalOptions = defaultOptions;\n\n    private subscriptions = new Subscription();\n    private cmpRef: ComponentRef<IModalDialog>;\n    private openTimer: any;\n    private modalHeightEvents$: Subject<void> = new Subject();\n\n    constructor(\n        private componentFactoryResolver: ComponentFactoryResolver,\n        private userAgent: UserAgentRef,\n        private renderer: Renderer2) {}\n\n    ngOnInit(): void {\n        this.isIE11 = this.userAgent.isIE11;\n\n        if (this.isIE11) {\n            this.subscriptions.add(\n                this.modalHeightEvents$.pipe(debounceTime(100)).subscribe(() => {\n                    this.ie11FixContentHeight();\n                })\n            );\n        }\n    }\n\n    ngOnDestroy(): void {\n        this.subscriptions.unsubscribe();\n        clearTimeout(this.openTimer);\n        if(this.cmpRef && this.cmpRef.destroy) {\n            this.cmpRef.destroy();\n        }\n    }\n\n    setOptions(options: IModalOptions): void {\n        this.options = Object.assign({}, defaultOptions, options);\n    }\n\n    /**\n     * Inject the component which will appear within the modal.\n     */\n    injectContent(component: Type<IModalDialog>): ComponentRef<IModalDialog> {\n        let factory = this.componentFactoryResolver.resolveComponentFactory(component);\n        this.cmpRef = this.portal.createComponent(factory);\n        return this.cmpRef;\n    }\n\n    /**\n     * Display the modal\n     */\n    open(): void {\n        clearTimeout(this.openTimer);\n        this.openTimer = setTimeout(() => this.visible = true, 50);\n    }\n\n    /**\n     * Programatically force the modal to close and resolve with the value passed.\n     */\n    forceClose(val?: any): void {\n        this.cmpRef.instance.closeFn(val);\n    }\n\n    /**\n     * Close the modal and call the cancelFn of the embedded component.\n     */\n    cancel(): void {\n        clearTimeout(this.openTimer);\n        this.visible = false;\n        this.cmpRef.instance.cancelFn();\n        this.cmpRef.destroy();\n    }\n\n    overlayClick(): void {\n        if (this.options.closeOnOverlayClick) {\n            this.cancel();\n        }\n    }\n\n    @HostListener('document:keydown', ['$event'])\n    keyHandler(e: KeyboardEvent): void {\n        if (e.which === 27 && this.options.closeOnEscape) {\n            this.cancel();\n        }\n    }\n\n    /**\n     * IE11 Related fixes\n     */\n\n    /**\n     * Listen for browsers size changes, to notify IE for modal height change\n     */\n    @HostListener('window:resize')\n    onResize(): void {\n        if (this.isIE11) {\n            this.modalHeightEvents$.next();\n        }\n    }\n\n    /**\n     * Listen for content changes, to notify IE for modal height change\n     */\n    ngAfterViewChecked(): void {\n        if (this.isIE11 && this.cmpRef) {\n            // Trigger modalHeight event on view checks\n            let modalElements = Array.from(this.cmpRef.location.nativeElement.children as HTMLElement[]);\n            let currentModalElementsHeight = modalElements\n            .filter( element => element.className != this.options.modalBodyClass )\n            .map( element => {\n                let styles = window.getComputedStyle(element);\n                let margin = parseFloat(styles['marginTop']) +\n                        parseFloat(styles['marginBottom']);\n                return element.offsetHeight + margin;\n            })\n            .reduce((heights: number, height: number): number => heights + height);\n\n            if (this.modalElementsHeight !== currentModalElementsHeight) {\n                this.modalElementsHeight = currentModalElementsHeight;\n                this.modalHeightEvents$.next();\n            }\n        }\n    }\n\n    /**\n     * Fixes modal body height for IE11\n     */\n    ie11FixContentHeight(): void {\n        if (this.isIE11 && this.cmpRef) {\n            let injectedElement = this.cmpRef.location.nativeElement as HTMLElement;\n            let modalBodyElement = injectedElement.getElementsByClassName(this.options.modalBodyClass)[0] as HTMLElement;\n\n            this.renderer.setStyle(\n                modalBodyElement,\n                'max-height',\n                `calc(70vh - ${this.modalElementsHeight}px)`\n            );\n        }\n    }\n}\n","<div class=\"gtx-modal-overlay\"\n     [class.visible]=\"visible\"\n     (click)=\"overlayClick()\"></div>\n<div class=\"gtx-modal-dialog\"\n     [class.isIE11]=\"isIE11\"\n     [class.visible]=\"visible\"\n     [style.width]=\"options.width\"\n     [class.nopad]=\"options.padding === false\">\n    <ng-template #portal></ng-template>\n</div>\n"]}