UNPKG

angular2-toaster

Version:

An Angular Toaster Notification library based on AngularJS-Toaster

158 lines 20.8 kB
import { Component, Input, Output, ViewChild, ViewContainerRef, EventEmitter, ComponentFactoryResolver, ChangeDetectorRef, HostListener, NgZone, ElementRef, Renderer2 } from '@angular/core'; import { BodyOutputType } from './bodyOutputType'; export class ToastComponent { constructor(componentFactoryResolver, changeDetectorRef, ngZone, element, renderer2) { this.componentFactoryResolver = componentFactoryResolver; this.changeDetectorRef = changeDetectorRef; this.ngZone = ngZone; this.element = element; this.renderer2 = renderer2; this.progressBarWidth = -1; this.bodyOutputType = BodyOutputType; this.clickEvent = new EventEmitter(); this.removeToastEvent = new EventEmitter(); this.timeoutId = null; this.timeout = 0; this.progressBarIntervalId = null; } ngOnInit() { if (this.toast.progressBar) { this.toast.progressBarDirection = this.toast.progressBarDirection || 'decreasing'; } let timeout = (typeof this.toast.timeout === 'number') ? this.toast.timeout : this.toasterconfig.timeout; if (typeof timeout === 'object') { timeout = timeout[this.toast.type]; } ; this.timeout = timeout; } ngAfterViewInit() { if (this.toast.bodyOutputType === this.bodyOutputType.Component) { const component = this.componentFactoryResolver.resolveComponentFactory(this.toast.body); const componentInstance = this.componentBody.createComponent(component, undefined, this.componentBody.injector); componentInstance.instance.toast = this.toast; this.changeDetectorRef.detectChanges(); } if (this.toasterconfig.mouseoverTimerStop) { // only apply a mouseenter event when necessary to avoid // unnecessary event and change detection cycles. this.removeMouseOverListener = this.renderer2.listen(this.element.nativeElement, 'mouseenter', () => this.stopTimer()); } this.configureTimer(); } click(event, toast) { event.stopPropagation(); this.clickEvent.emit({ value: { toast: toast, isCloseButton: true } }); } stopTimer() { this.progressBarWidth = 0; this.clearTimers(); } restartTimer() { if (this.toasterconfig.mouseoverTimerStop) { if (!this.timeoutId) { this.configureTimer(); } } else if (this.timeout && !this.timeoutId) { this.removeToast(); } } ngOnDestroy() { if (this.removeMouseOverListener) { this.removeMouseOverListener(); } this.clearTimers(); } configureTimer() { if (!this.timeout || this.timeout < 1) { return; } if (this.toast.progressBar) { this.removeToastTick = new Date().getTime() + this.timeout; this.progressBarWidth = -1; } this.ngZone.runOutsideAngular(() => { this.timeoutId = window.setTimeout(() => { this.ngZone.run(() => { this.changeDetectorRef.markForCheck(); this.removeToast(); }); }, this.timeout); if (this.toast.progressBar) { this.progressBarIntervalId = window.setInterval(() => { this.ngZone.run(() => { this.updateProgressBar(); }); }, 10); } }); } updateProgressBar() { if (this.progressBarWidth === 0 || this.progressBarWidth === 100) { return; } this.progressBarWidth = ((this.removeToastTick - new Date().getTime()) / this.timeout) * 100; if (this.toast.progressBarDirection === 'increasing') { this.progressBarWidth = 100 - this.progressBarWidth; } if (this.progressBarWidth < 0) { this.progressBarWidth = 0; } if (this.progressBarWidth > 100) { this.progressBarWidth = 100; } } clearTimers() { if (this.timeoutId) { window.clearTimeout(this.timeoutId); } if (this.progressBarIntervalId) { window.clearInterval(this.progressBarIntervalId); } this.timeoutId = null; this.progressBarIntervalId = null; } removeToast() { this.removeToastEvent.emit(this.toast); } } ToastComponent.decorators = [ { type: Component, args: [{ selector: '[toastComp]', template: ` <div class="toast-content"> <div [ngClass]="titleClass">{{toast.title}}</div> <div [ngClass]="messageClass" [ngSwitch]="toast.bodyOutputType"> <div *ngSwitchCase="bodyOutputType.Component" #componentBody></div> <div *ngSwitchCase="bodyOutputType.TrustedHtml" [innerHTML]="toast.body | trustHtml"></div> <div *ngSwitchCase="bodyOutputType.Default">{{toast.body}}</div> </div> </div> <button class="toast-close-button" *ngIf="toast.showCloseButton" (click)="click($event, toast)" [innerHTML]="toast.closeHtml | trustHtml"> </button> <div *ngIf="toast.progressBar"> <div class="toast-progress-bar" [style.width]="progressBarWidth + '%'"></div> </div>` },] } ]; ToastComponent.ctorParameters = () => [ { type: ComponentFactoryResolver }, { type: ChangeDetectorRef }, { type: NgZone }, { type: ElementRef }, { type: Renderer2 } ]; ToastComponent.propDecorators = { toasterconfig: [{ type: Input }], toast: [{ type: Input }], titleClass: [{ type: Input }], messageClass: [{ type: Input }], componentBody: [{ type: ViewChild, args: ['componentBody', { read: ViewContainerRef, static: false },] }], clickEvent: [{ type: Output }], removeToastEvent: [{ type: Output }], restartTimer: [{ type: HostListener, args: ['mouseleave',] }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toast.component.js","sourceRoot":"../../../../src/angular2-toaster/src/","sources":["lib/toast.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component, \n    Input, \n    Output, \n    ViewChild, \n    ViewContainerRef, \n    EventEmitter,\n    ComponentFactoryResolver, \n    ChangeDetectorRef, \n    OnInit, \n    AfterViewInit, \n    OnDestroy,\n    HostListener,\n    NgZone, \n    ElementRef,\n    Renderer2\n} from '@angular/core';\nimport { Toast } from './toast';\nimport { BodyOutputType } from './bodyOutputType';\nimport { ToasterConfig } from './toaster-config';\n\n@Component({\n    selector: '[toastComp]',\n    template: `\n        <div class=\"toast-content\">\n            <div [ngClass]=\"titleClass\">{{toast.title}}</div>\n            <div [ngClass]=\"messageClass\" [ngSwitch]=\"toast.bodyOutputType\">\n                <div *ngSwitchCase=\"bodyOutputType.Component\" #componentBody></div>\n                <div *ngSwitchCase=\"bodyOutputType.TrustedHtml\" [innerHTML]=\"toast.body | trustHtml\"></div>\n                <div *ngSwitchCase=\"bodyOutputType.Default\">{{toast.body}}</div>\n            </div>\n        </div>\n        <button class=\"toast-close-button\" *ngIf=\"toast.showCloseButton\" (click)=\"click($event, toast)\"\n            [innerHTML]=\"toast.closeHtml | trustHtml\">\n        </button>\n        <div *ngIf=\"toast.progressBar\">\n            <div class=\"toast-progress-bar\" [style.width]=\"progressBarWidth + '%'\"></div>\n        </div>`\n})\nexport class ToastComponent implements OnInit, AfterViewInit, OnDestroy {\n    @Input() toasterconfig: ToasterConfig;\n    @Input() toast: Toast;\n    @Input() titleClass: string;\n    @Input() messageClass: string;\n    @ViewChild('componentBody', { read: ViewContainerRef, static: false }) componentBody: ViewContainerRef;\n\n    public progressBarWidth: number = -1;\n    public bodyOutputType = BodyOutputType;\n\n    @Output()\n    public clickEvent = new EventEmitter();\n    @Output()\n    public removeToastEvent = new EventEmitter<Toast>();\n\n    private timeoutId?: number = null;\n    private timeout: number = 0;\n    private progressBarIntervalId?: number = null;\n    private removeToastTick: number;\n\n    private removeMouseOverListener: () => void;\n\n    constructor(\n      private componentFactoryResolver: ComponentFactoryResolver,\n      private changeDetectorRef: ChangeDetectorRef,\n      private ngZone: NgZone,\n      private element: ElementRef,\n      private renderer2: Renderer2\n    ) {}\n\n    ngOnInit() {\n        if (this.toast.progressBar) {\n            this.toast.progressBarDirection = this.toast.progressBarDirection || 'decreasing';\n        }\n\n        let timeout = (typeof this.toast.timeout === 'number')\n            ? this.toast.timeout : this.toasterconfig.timeout;\n\n        if (typeof timeout === 'object') { \n            timeout = timeout[this.toast.type]; \n        };\n\n        this.timeout = timeout;\n    }\n\n    ngAfterViewInit() {\n        if (this.toast.bodyOutputType === this.bodyOutputType.Component) {\n            const component = this.componentFactoryResolver.resolveComponentFactory(this.toast.body);\n            const componentInstance: any = this.componentBody.createComponent(component, undefined, this.componentBody.injector);\n            componentInstance.instance.toast = this.toast;\n            this.changeDetectorRef.detectChanges();\n        }\n\n        if (this.toasterconfig.mouseoverTimerStop) {\n            // only apply a mouseenter event when necessary to avoid\n            // unnecessary event and change detection cycles.\n            this.removeMouseOverListener = this.renderer2.listen(\n                this.element.nativeElement, \n                'mouseenter', \n                () => this.stopTimer()\n            );\n        }\n\n        this.configureTimer();\n    }\n\n    click(event: MouseEvent, toast: Toast) {\n        event.stopPropagation();\n        this.clickEvent.emit({ value : { toast: toast, isCloseButton: true } });\n    }\n\n    stopTimer() {\n        this.progressBarWidth = 0;\n        this.clearTimers();\n    }\n\n    @HostListener('mouseleave') \n    restartTimer() {\n        if (this.toasterconfig.mouseoverTimerStop) {\n            if (!this.timeoutId) {\n                this.configureTimer();\n            }\n        } else if (this.timeout && !this.timeoutId) {\n            this.removeToast();\n        }\n    }\n\n    ngOnDestroy() {\n        if (this.removeMouseOverListener) {\n            this.removeMouseOverListener();\n        }\n        this.clearTimers();\n    }\n\n    private configureTimer() {\n        if (!this.timeout || this.timeout < 1) {\n            return;\n        }\n\n        if (this.toast.progressBar) {\n            this.removeToastTick = new Date().getTime() + this.timeout;\n            this.progressBarWidth = -1;\n        } \n        \n        this.ngZone.runOutsideAngular(() => {\n            this.timeoutId = window.setTimeout(() => {\n                this.ngZone.run(() => {\n                    this.changeDetectorRef.markForCheck();\n                    this.removeToast();\n                });\n            }, this.timeout);\n\n            if (this.toast.progressBar) {\n                this.progressBarIntervalId = window.setInterval(() => {\n                    this.ngZone.run(() => {\n                        this.updateProgressBar();\n                    });\n                }, 10);\n            }\n        });\n    }\n\n    private updateProgressBar() {\n        if (this.progressBarWidth === 0 || this.progressBarWidth === 100) {\n          return;\n        }\n\n        this.progressBarWidth = ((this.removeToastTick - new Date().getTime()) / this.timeout) * 100;\n        \n        if (this.toast.progressBarDirection === 'increasing') {\n          this.progressBarWidth = 100 - this.progressBarWidth;\n        }\n        if (this.progressBarWidth < 0) {\n          this.progressBarWidth = 0;\n        }\n        if (this.progressBarWidth > 100) {\n          this.progressBarWidth = 100;\n        }\n    }\n\n    private clearTimers() {\n        if (this.timeoutId) {\n            window.clearTimeout(this.timeoutId)\n        }\n\n        if (this.progressBarIntervalId) {\n            window.clearInterval(this.progressBarIntervalId);\n        }\n\n        this.timeoutId = null;\n        this.progressBarIntervalId = null;\n    }\n\n    private removeToast() {\n        this.removeToastEvent.emit(this.toast);\n    }\n}\n"]}