angular2-toaster
Version:
An Angular Toaster Notification library based on AngularJS-Toaster
158 lines • 20.8 kB
JavaScript
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"]}