UNPKG

@danielmoncada/angular-datetime-picker

Version:
208 lines 26.6 kB
/** * dialog-container.component */ import { ChangeDetectorRef, Component, ElementRef, EventEmitter, Inject, Optional, ViewChild } from '@angular/core'; import { animate, animateChild, keyframes, style, transition, trigger } from '@angular/animations'; import { DOCUMENT } from '@angular/common'; import { FocusTrapFactory } from '@angular/cdk/a11y'; import { BasePortalOutlet, CdkPortalOutlet } from '@angular/cdk/portal'; const zoomFadeIn = { opacity: 0, transform: 'translateX({{ x }}) translateY({{ y }}) scale({{scale}})' }; const zoomFadeInFrom = { opacity: 0, transform: 'translateX({{ x }}) translateY({{ y }}) scale({{scale}})', transformOrigin: '{{ ox }} {{ oy }}' }; export class OwlDialogContainerComponent extends BasePortalOutlet { constructor(changeDetector, elementRef, focusTrapFactory, document) { super(); this.changeDetector = changeDetector; this.elementRef = elementRef; this.focusTrapFactory = focusTrapFactory; this.document = document; this.portalOutlet = null; /** ID of the element that should be considered as the dialog's label. */ this.ariaLabelledBy = null; /** Emits when an animation state changes. */ this.animationStateChanged = new EventEmitter(); this.isAnimating = false; this.state = 'enter'; // for animation purpose this.params = { x: '0px', y: '0px', ox: '50%', oy: '50%', scale: 0 }; // A variable to hold the focused element before the dialog was open. // This would help us to refocus back to element when the dialog was closed. this.elementFocusedBeforeDialogWasOpened = null; } get config() { return this._config; } get owlDialogContainerClass() { return true; } get owlDialogContainerTabIndex() { return -1; } get owlDialogContainerId() { return this._config.id; } get owlDialogContainerRole() { return this._config.role || null; } get owlDialogContainerAriaLabelledby() { return this.ariaLabelledBy; } get owlDialogContainerAriaDescribedby() { return this._config.ariaDescribedBy || null; } get owlDialogContainerAnimation() { return { value: this.state, params: this.params }; } ngOnInit() { } /** * Attach a ComponentPortal as content to this dialog container. */ attachComponentPortal(portal) { if (this.portalOutlet.hasAttached()) { throw Error('Attempting to attach dialog content after content is already attached'); } this.savePreviouslyFocusedElement(); return this.portalOutlet.attachComponentPortal(portal); } attachTemplatePortal(portal) { throw new Error('Method not implemented.'); } setConfig(config) { this._config = config; if (config.event) { this.calculateZoomOrigin(event); } } onAnimationStart(event) { this.isAnimating = true; this.animationStateChanged.emit(event); } onAnimationDone(event) { if (event.toState === 'enter') { this.trapFocus(); } else if (event.toState === 'exit') { this.restoreFocus(); } this.animationStateChanged.emit(event); this.isAnimating = false; } startExitAnimation() { this.state = 'exit'; this.changeDetector.markForCheck(); } /** * Calculate origin used in the `zoomFadeInFrom()` * for animation purpose */ calculateZoomOrigin(event) { if (!event) { return; } const clientX = event.clientX; const clientY = event.clientY; const wh = window.innerWidth / 2; const hh = window.innerHeight / 2; const x = clientX - wh; const y = clientY - hh; const ox = clientX / window.innerWidth; const oy = clientY / window.innerHeight; this.params.x = `${x}px`; this.params.y = `${y}px`; this.params.ox = `${ox * 100}%`; this.params.oy = `${oy * 100}%`; this.params.scale = 0; return; } /** * Save the focused element before dialog was open */ savePreviouslyFocusedElement() { if (this.document) { this.elementFocusedBeforeDialogWasOpened = this.document .activeElement; Promise.resolve().then(() => this.elementRef.nativeElement.focus()); } } trapFocus() { if (!this.focusTrap) { this.focusTrap = this.focusTrapFactory.create(this.elementRef.nativeElement); } if (this._config.autoFocus) { this.focusTrap.focusInitialElementWhenReady(); } } restoreFocus() { const toFocus = this.elementFocusedBeforeDialogWasOpened; // We need the extra check, because IE can set the `activeElement` to null in some cases. if (toFocus && typeof toFocus.focus === 'function') { toFocus.focus(); } if (this.focusTrap) { this.focusTrap.destroy(); } } } OwlDialogContainerComponent.decorators = [ { type: Component, args: [{ selector: 'owl-dialog-container', template: "<ng-template [cdkPortalOutlet]></ng-template>\n", animations: [ trigger('slideModal', [ transition('void => enter', [ style(zoomFadeInFrom), animate('300ms cubic-bezier(0.35, 0, 0.25, 1)', style('*')), animate('150ms', keyframes([ style({ transform: 'scale(1)', offset: 0 }), style({ transform: 'scale(1.05)', offset: 0.3 }), style({ transform: 'scale(.95)', offset: 0.8 }), style({ transform: 'scale(1)', offset: 1.0 }) ])), animateChild() ], { params: { x: '0px', y: '0px', ox: '50%', oy: '50%', scale: 1 } }), transition('enter => exit', [animateChild(), animate(200, style(zoomFadeIn))], { params: { x: '0px', y: '0px', ox: '50%', oy: '50%' } }) ]) ], host: { '(@slideModal.start)': 'onAnimationStart($event)', '(@slideModal.done)': 'onAnimationDone($event)', '[class.owl-dialog-container]': 'owlDialogContainerClass', '[attr.tabindex]': 'owlDialogContainerTabIndex', '[attr.id]': 'owlDialogContainerId', '[attr.role]': 'owlDialogContainerRole', '[attr.aria-labelledby]': 'owlDialogContainerAriaLabelledby', '[attr.aria-describedby]': 'owlDialogContainerAriaDescribedby', '[@slideModal]': 'owlDialogContainerAnimation' } },] } ]; OwlDialogContainerComponent.ctorParameters = () => [ { type: ChangeDetectorRef }, { type: ElementRef }, { type: FocusTrapFactory }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] } ]; OwlDialogContainerComponent.propDecorators = { portalOutlet: [{ type: ViewChild, args: [CdkPortalOutlet, { static: true },] }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog-container.component.js","sourceRoot":"","sources":["../../../../../projects/picker/src/lib/dialog/dialog-container.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * dialog-container.component\n */\n\nimport {\n    ChangeDetectorRef,\n    Component,\n    ComponentRef,\n    ElementRef,\n    EmbeddedViewRef,\n    EventEmitter,\n    Inject,\n    OnInit,\n    Optional,\n    ViewChild\n} from '@angular/core';\nimport {\n    animate,\n    animateChild,\n    AnimationEvent,\n    keyframes,\n    style,\n    transition,\n    trigger\n} from '@angular/animations';\nimport { DOCUMENT } from '@angular/common';\nimport { FocusTrap, FocusTrapFactory } from '@angular/cdk/a11y';\nimport {\n    BasePortalOutlet,\n    CdkPortalOutlet,\n    ComponentPortal,\n    TemplatePortal\n} from '@angular/cdk/portal';\nimport { OwlDialogConfigInterface } from './dialog-config.class';\n\nconst zoomFadeIn = {\n    opacity: 0,\n    transform: 'translateX({{ x }}) translateY({{ y }}) scale({{scale}})'\n};\nconst zoomFadeInFrom = {\n    opacity: 0,\n    transform: 'translateX({{ x }}) translateY({{ y }}) scale({{scale}})',\n    transformOrigin: '{{ ox }} {{ oy }}'\n};\n\n@Component({\n    selector: 'owl-dialog-container',\n    templateUrl: './dialog-container.component.html',\n    animations: [\n        trigger('slideModal', [\n            transition(\n                'void => enter',\n                [\n                    style(zoomFadeInFrom),\n                    animate('300ms cubic-bezier(0.35, 0, 0.25, 1)', style('*')),\n                    animate(\n                        '150ms',\n                        keyframes([\n                            style({ transform: 'scale(1)', offset: 0 }),\n                            style({ transform: 'scale(1.05)', offset: 0.3 }),\n                            style({ transform: 'scale(.95)', offset: 0.8 }),\n                            style({ transform: 'scale(1)', offset: 1.0 })\n                        ])\n                    ),\n                    animateChild()\n                ],\n                {\n                    params: {\n                        x: '0px',\n                        y: '0px',\n                        ox: '50%',\n                        oy: '50%',\n                        scale: 1\n                    }\n                }\n            ),\n            transition(\n                'enter => exit',\n                [animateChild(), animate(200, style(zoomFadeIn))],\n                { params: { x: '0px', y: '0px', ox: '50%', oy: '50%' } }\n            )\n        ])\n    ],\n    host: {\n        '(@slideModal.start)': 'onAnimationStart($event)',\n        '(@slideModal.done)': 'onAnimationDone($event)',\n        '[class.owl-dialog-container]': 'owlDialogContainerClass',\n        '[attr.tabindex]': 'owlDialogContainerTabIndex',\n        '[attr.id]': 'owlDialogContainerId',\n        '[attr.role]': 'owlDialogContainerRole',\n        '[attr.aria-labelledby]': 'owlDialogContainerAriaLabelledby',\n        '[attr.aria-describedby]': 'owlDialogContainerAriaDescribedby',\n        '[@slideModal]': 'owlDialogContainerAnimation'\n    }\n})\nexport class OwlDialogContainerComponent extends BasePortalOutlet\n    implements OnInit {\n    @ViewChild(CdkPortalOutlet, { static: true })\n    portalOutlet: CdkPortalOutlet | null = null;\n\n    /** The class that traps and manages focus within the dialog. */\n    private focusTrap: FocusTrap;\n\n    /** ID of the element that should be considered as the dialog's label. */\n    public ariaLabelledBy: string | null = null;\n\n    /** Emits when an animation state changes. */\n    public animationStateChanged = new EventEmitter<AnimationEvent>();\n\n    public isAnimating = false;\n\n    private _config: OwlDialogConfigInterface;\n    get config(): OwlDialogConfigInterface {\n        return this._config;\n    }\n\n    private state: 'void' | 'enter' | 'exit' = 'enter';\n\n    // for animation purpose\n    private params: any = {\n        x: '0px',\n        y: '0px',\n        ox: '50%',\n        oy: '50%',\n        scale: 0\n    };\n\n    // A variable to hold the focused element before the dialog was open.\n    // This would help us to refocus back to element when the dialog was closed.\n    private elementFocusedBeforeDialogWasOpened: HTMLElement | null = null;\n\n    get owlDialogContainerClass(): boolean {\n        return true;\n    }\n\n    get owlDialogContainerTabIndex(): number {\n        return -1;\n    }\n\n    get owlDialogContainerId(): string {\n        return this._config.id;\n    }\n\n    get owlDialogContainerRole(): string {\n        return this._config.role || null;\n    }\n\n    get owlDialogContainerAriaLabelledby(): string {\n        return this.ariaLabelledBy;\n    }\n\n    get owlDialogContainerAriaDescribedby(): string {\n        return this._config.ariaDescribedBy || null;\n    }\n\n    get owlDialogContainerAnimation(): any {\n        return { value: this.state, params: this.params };\n    }\n\n    constructor(\n        private changeDetector: ChangeDetectorRef,\n        private elementRef: ElementRef,\n        private focusTrapFactory: FocusTrapFactory,\n        @Optional()\n        @Inject(DOCUMENT)\n        private document: any\n    ) {\n        super();\n    }\n\n    public ngOnInit() {}\n\n    /**\n     * Attach a ComponentPortal as content to this dialog container.\n     */\n    public attachComponentPortal<T>(\n        portal: ComponentPortal<T>\n    ): ComponentRef<T> {\n        if (this.portalOutlet.hasAttached()) {\n            throw Error(\n                'Attempting to attach dialog content after content is already attached'\n            );\n        }\n\n        this.savePreviouslyFocusedElement();\n        return this.portalOutlet.attachComponentPortal(portal);\n    }\n\n    public attachTemplatePortal<C>(\n        portal: TemplatePortal<C>\n    ): EmbeddedViewRef<C> {\n        throw new Error('Method not implemented.');\n    }\n\n    public setConfig(config: OwlDialogConfigInterface): void {\n        this._config = config;\n\n        if (config.event) {\n            this.calculateZoomOrigin(event);\n        }\n    }\n\n    public onAnimationStart( event: AnimationEvent ): void {\n        this.isAnimating = true;\n        this.animationStateChanged.emit(event);\n    }\n\n    public onAnimationDone( event: AnimationEvent ): void {\n        if (event.toState === 'enter') {\n            this.trapFocus();\n        } else if (event.toState === 'exit') {\n            this.restoreFocus();\n        }\n\n        this.animationStateChanged.emit(event);\n        this.isAnimating = false;\n    }\n\n    public startExitAnimation() {\n        this.state = 'exit';\n        this.changeDetector.markForCheck();\n    }\n\n    /**\n     * Calculate origin used in the `zoomFadeInFrom()`\n     * for animation purpose\n     */\n    private calculateZoomOrigin(event: any): void {\n        if (!event) {\n            return;\n        }\n\n        const clientX = event.clientX;\n        const clientY = event.clientY;\n\n        const wh = window.innerWidth / 2;\n        const hh = window.innerHeight / 2;\n        const x = clientX - wh;\n        const y = clientY - hh;\n        const ox = clientX / window.innerWidth;\n        const oy = clientY / window.innerHeight;\n\n        this.params.x = `${x}px`;\n        this.params.y = `${y}px`;\n        this.params.ox = `${ox * 100}%`;\n        this.params.oy = `${oy * 100}%`;\n        this.params.scale = 0;\n\n        return;\n    }\n\n    /**\n     * Save the focused element before dialog was open\n     */\n    private savePreviouslyFocusedElement(): void {\n        if (this.document) {\n            this.elementFocusedBeforeDialogWasOpened = this.document\n                .activeElement as HTMLElement;\n\n            Promise.resolve().then(() => this.elementRef.nativeElement.focus());\n        }\n    }\n\n    private trapFocus(): void {\n        if (!this.focusTrap) {\n            this.focusTrap = this.focusTrapFactory.create(\n                this.elementRef.nativeElement\n            );\n        }\n\n        if (this._config.autoFocus) {\n            this.focusTrap.focusInitialElementWhenReady();\n        }\n    }\n\n    private restoreFocus(): void {\n        const toFocus = this.elementFocusedBeforeDialogWasOpened;\n\n        // We need the extra check, because IE can set the `activeElement` to null in some cases.\n        if (toFocus && typeof toFocus.focus === 'function') {\n            toFocus.focus();\n        }\n\n        if (this.focusTrap) {\n            this.focusTrap.destroy();\n        }\n    }\n}\n"]}