UNPKG

ng-zorro-antd

Version:

An enterprise-class UI components based on Ant Design and Angular

588 lines (577 loc) 22.5 kB
import { __decorate, __metadata, __rest } from 'tslib'; import { FocusTrapFactory } from '@angular/cdk/a11y'; import { Directionality, BidiModule } from '@angular/cdk/bidi'; import { ESCAPE } from '@angular/cdk/keycodes'; import { OverlayConfig, Overlay, OverlayKeyboardDispatcher, OverlayModule } from '@angular/cdk/overlay'; import { ComponentPortal, TemplatePortal, CdkPortalOutlet, PortalModule } from '@angular/cdk/portal'; import { DOCUMENT, CommonModule } from '@angular/common'; import { Directive, TemplateRef, EventEmitter, Type, Injector, Component, ChangeDetectionStrategy, ChangeDetectorRef, Optional, Inject, Renderer2, ViewContainerRef, Input, Output, ViewChild, ContentChild, NgModule, ɵɵdefineInjectable, ɵɵinject, Injectable } from '@angular/core'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { NzConfigService, WithConfig } from 'ng-zorro-antd/core/config'; import { warnDeprecation } from 'ng-zorro-antd/core/logger'; import { toCssPixel, InputBoolean } from 'ng-zorro-antd/core/util'; import { NzNoAnimationModule } from 'ng-zorro-antd/core/no-animation'; import { NzOutletModule } from 'ng-zorro-antd/core/outlet'; import { NzIconModule } from 'ng-zorro-antd/icon'; /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ class NzDrawerContentDirective { constructor(templateRef) { this.templateRef = templateRef; } } NzDrawerContentDirective.decorators = [ { type: Directive, args: [{ selector: '[nzDrawerContent]', exportAs: 'nzDrawerContent' },] } ]; NzDrawerContentDirective.ctorParameters = () => [ { type: TemplateRef } ]; /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ class NzDrawerRef { } /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ const DRAWER_ANIMATE_DURATION = 300; const NZ_CONFIG_MODULE_NAME = 'drawer'; class NzDrawerComponent extends NzDrawerRef { constructor(cdr, // tslint:disable-next-line:no-any document, nzConfigService, renderer, overlay, injector, changeDetectorRef, focusTrapFactory, viewContainerRef, overlayKeyboardDispatcher, directionality) { super(); this.cdr = cdr; this.document = document; this.nzConfigService = nzConfigService; this.renderer = renderer; this.overlay = overlay; this.injector = injector; this.changeDetectorRef = changeDetectorRef; this.focusTrapFactory = focusTrapFactory; this.viewContainerRef = viewContainerRef; this.overlayKeyboardDispatcher = overlayKeyboardDispatcher; this.directionality = directionality; this._nzModuleName = NZ_CONFIG_MODULE_NAME; this.nzCloseIcon = 'close'; this.nzClosable = true; this.nzMaskClosable = true; this.nzMask = true; this.nzCloseOnNavigation = true; this.nzNoAnimation = false; this.nzKeyboard = true; this.nzPlacement = 'right'; this.nzMaskStyle = {}; this.nzBodyStyle = {}; this.nzWidth = 256; this.nzHeight = 256; this.nzZIndex = 1000; this.nzOffsetX = 0; this.nzOffsetY = 0; this.componentInstance = null; this.nzOnViewInit = new EventEmitter(); this.nzOnClose = new EventEmitter(); this.nzVisibleChange = new EventEmitter(); this.destroy$ = new Subject(); this.placementChanging = false; this.placementChangeTimeoutId = -1; this.isOpen = false; this.templateContext = { $implicit: undefined, drawerRef: this }; this.nzAfterOpen = new Subject(); this.nzAfterClose = new Subject(); // from service config this.nzDirection = undefined; this.dir = 'ltr'; } set nzVisible(value) { this.isOpen = value; } get nzVisible() { return this.isOpen; } get offsetTransform() { if (!this.isOpen || this.nzOffsetX + this.nzOffsetY === 0) { return null; } switch (this.nzPlacement) { case 'left': return `translateX(${this.nzOffsetX}px)`; case 'right': return `translateX(-${this.nzOffsetX}px)`; case 'top': return `translateY(${this.nzOffsetY}px)`; case 'bottom': return `translateY(-${this.nzOffsetY}px)`; } } get transform() { if (this.isOpen) { return null; } switch (this.nzPlacement) { case 'left': return `translateX(-100%)`; case 'right': return `translateX(100%)`; case 'top': return `translateY(-100%)`; case 'bottom': return `translateY(100%)`; } } get width() { return this.isLeftOrRight ? toCssPixel(this.nzWidth) : null; } get height() { return !this.isLeftOrRight ? toCssPixel(this.nzHeight) : null; } get isLeftOrRight() { return this.nzPlacement === 'left' || this.nzPlacement === 'right'; } get afterOpen() { return this.nzAfterOpen.asObservable(); } get afterClose() { return this.nzAfterClose.asObservable(); } isTemplateRef(value) { return value instanceof TemplateRef; } ngOnInit() { var _a; (_a = this.directionality.change) === null || _a === void 0 ? void 0 : _a.pipe(takeUntil(this.destroy$)).subscribe((direction) => { this.dir = direction; this.cdr.detectChanges(); }); this.dir = this.nzDirection || this.directionality.value; this.attachOverlay(); this.updateOverlayStyle(); this.updateBodyOverflow(); this.templateContext = { $implicit: this.nzContentParams, drawerRef: this }; this.changeDetectorRef.detectChanges(); } ngAfterViewInit() { this.attachBodyContent(); setTimeout(() => { this.nzOnViewInit.emit(); }); } ngAfterContentInit() { if (!(this.contentFromContentChild || this.nzContent)) { warnDeprecation('Usage `<ng-content></ng-content>` is deprecated, which will be removed in 12.0.0. Please instead use `<ng-template nzDrawerContent></ng-template>` to declare the content of the drawer.'); } } ngOnChanges(changes) { const { nzPlacement, nzVisible } = changes; if (nzVisible) { const value = changes.nzVisible.currentValue; if (value) { this.open(); } else { this.close(); } } if (nzPlacement && !nzPlacement.isFirstChange()) { this.triggerPlacementChangeCycleOnce(); } } ngOnDestroy() { this.destroy$.next(); this.destroy$.complete(); clearTimeout(this.placementChangeTimeoutId); this.disposeOverlay(); } getAnimationDuration() { return this.nzNoAnimation ? 0 : DRAWER_ANIMATE_DURATION; } // Disable the transition animation temporarily when the placement changing triggerPlacementChangeCycleOnce() { if (!this.nzNoAnimation) { this.placementChanging = true; this.changeDetectorRef.markForCheck(); clearTimeout(this.placementChangeTimeoutId); this.placementChangeTimeoutId = setTimeout(() => { this.placementChanging = false; this.changeDetectorRef.markForCheck(); }, this.getAnimationDuration()); } } close(result) { this.isOpen = false; this.nzVisibleChange.emit(false); this.updateOverlayStyle(); this.overlayKeyboardDispatcher.remove(this.overlayRef); this.changeDetectorRef.detectChanges(); setTimeout(() => { this.updateBodyOverflow(); this.restoreFocus(); this.nzAfterClose.next(result); this.nzAfterClose.complete(); this.componentInstance = null; }, this.getAnimationDuration()); } open() { this.attachOverlay(); this.isOpen = true; this.nzVisibleChange.emit(true); this.overlayKeyboardDispatcher.add(this.overlayRef); this.updateOverlayStyle(); this.updateBodyOverflow(); this.savePreviouslyFocusedElement(); this.trapFocus(); this.changeDetectorRef.detectChanges(); setTimeout(() => { this.nzAfterOpen.next(); }, this.getAnimationDuration()); } getContentComponent() { return this.componentInstance; } closeClick() { this.nzOnClose.emit(); } maskClick() { if (this.nzMaskClosable && this.nzMask) { this.nzOnClose.emit(); } } attachBodyContent() { this.bodyPortalOutlet.dispose(); if (this.nzContent instanceof Type) { const childInjector = Injector.create({ parent: this.injector, providers: [{ provide: NzDrawerRef, useValue: this }] }); const componentPortal = new ComponentPortal(this.nzContent, null, childInjector); const componentRef = this.bodyPortalOutlet.attachComponentPortal(componentPortal); this.componentInstance = componentRef.instance; Object.assign(componentRef.instance, this.nzContentParams); componentRef.changeDetectorRef.detectChanges(); } } attachOverlay() { if (!this.overlayRef) { this.portal = new TemplatePortal(this.drawerTemplate, this.viewContainerRef); this.overlayRef = this.overlay.create(this.getOverlayConfig()); } if (this.overlayRef && !this.overlayRef.hasAttached()) { this.overlayRef.attach(this.portal); this.overlayRef.keydownEvents() .pipe(takeUntil(this.destroy$)) .subscribe((event) => { if (event.keyCode === ESCAPE && this.isOpen && this.nzKeyboard) { this.nzOnClose.emit(); } }); this.overlayRef .detachments() .pipe(takeUntil(this.destroy$)) .subscribe(() => { this.disposeOverlay(); }); } } disposeOverlay() { var _a; (_a = this.overlayRef) === null || _a === void 0 ? void 0 : _a.dispose(); this.overlayRef = null; } getOverlayConfig() { return new OverlayConfig({ disposeOnNavigation: this.nzCloseOnNavigation, positionStrategy: this.overlay.position().global(), scrollStrategy: this.overlay.scrollStrategies.block() }); } updateOverlayStyle() { if (this.overlayRef && this.overlayRef.overlayElement) { this.renderer.setStyle(this.overlayRef.overlayElement, 'pointer-events', this.isOpen ? 'auto' : 'none'); } } updateBodyOverflow() { if (this.overlayRef) { if (this.isOpen) { this.overlayRef.getConfig().scrollStrategy.enable(); } else { this.overlayRef.getConfig().scrollStrategy.disable(); } } } savePreviouslyFocusedElement() { if (this.document && !this.previouslyFocusedElement) { this.previouslyFocusedElement = this.document.activeElement; // We need the extra check, because IE's svg element has no blur method. if (this.previouslyFocusedElement && typeof this.previouslyFocusedElement.blur === 'function') { this.previouslyFocusedElement.blur(); } } } trapFocus() { if (!this.focusTrap && this.overlayRef && this.overlayRef.overlayElement) { this.focusTrap = this.focusTrapFactory.create(this.overlayRef.overlayElement); this.focusTrap.focusInitialElement(); } } restoreFocus() { // We need the extra check, because IE can set the `activeElement` to null in some cases. if (this.previouslyFocusedElement && typeof this.previouslyFocusedElement.focus === 'function') { this.previouslyFocusedElement.focus(); } if (this.focusTrap) { this.focusTrap.destroy(); } } } NzDrawerComponent.decorators = [ { type: Component, args: [{ selector: 'nz-drawer', exportAs: 'nzDrawer', template: ` <ng-template #drawerTemplate> <div class="ant-drawer" [nzNoAnimation]="nzNoAnimation" [class.ant-drawer-rtl]="dir === 'rtl'" [class.ant-drawer-open]="isOpen" [class.no-mask]="!nzMask" [class.ant-drawer-top]="nzPlacement === 'top'" [class.ant-drawer-bottom]="nzPlacement === 'bottom'" [class.ant-drawer-right]="nzPlacement === 'right'" [class.ant-drawer-left]="nzPlacement === 'left'" [style.transform]="offsetTransform" [style.transition]="placementChanging ? 'none' : null" [style.zIndex]="nzZIndex" > <div class="ant-drawer-mask" (click)="maskClick()" *ngIf="nzMask" [ngStyle]="nzMaskStyle"></div> <div class="ant-drawer-content-wrapper {{ nzWrapClassName }}" [style.width]="width" [style.height]="height" [style.transform]="transform" [style.transition]="placementChanging ? 'none' : null" > <div class="ant-drawer-content"> <div class="ant-drawer-wrapper-body" [style.height]="isLeftOrRight ? '100%' : null"> <div *ngIf="nzTitle || nzClosable" [class.ant-drawer-header]="!!nzTitle" [class.ant-drawer-header-no-title]="!nzTitle"> <div *ngIf="nzTitle" class="ant-drawer-title"> <ng-container *nzStringTemplateOutlet="nzTitle"><div [innerHTML]="nzTitle"></div></ng-container> </div> <button *ngIf="nzClosable" (click)="closeClick()" aria-label="Close" class="ant-drawer-close" style="--scroll-bar: 0px;"> <ng-container *nzStringTemplateOutlet="nzCloseIcon; let closeIcon"> <i nz-icon [nzType]="closeIcon"></i> </ng-container> </button> </div> <div class="ant-drawer-body" [ngStyle]="nzBodyStyle"> <ng-template cdkPortalOutlet></ng-template> <ng-container *ngIf="nzContent; else contentElseTemp"> <ng-container *ngIf="isTemplateRef(nzContent)"> <ng-container *ngTemplateOutlet="$any(nzContent); context: templateContext"></ng-container> </ng-container> </ng-container> <ng-template #contentElseTemp> <ng-container *ngIf="contentFromContentChild"> <ng-template [ngTemplateOutlet]="contentFromContentChild"></ng-template> </ng-container> </ng-template> <ng-content *ngIf="!(nzContent || contentFromContentChild)"></ng-content> </div> <div *ngIf="nzFooter" class="ant-drawer-footer"> <ng-container *nzStringTemplateOutlet="nzFooter"><div [innerHTML]="nzFooter"></div></ng-container> </div> </div> </div> </div> </div> </ng-template> `, preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush },] } ]; NzDrawerComponent.ctorParameters = () => [ { type: ChangeDetectorRef }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] }, { type: NzConfigService }, { type: Renderer2 }, { type: Overlay }, { type: Injector }, { type: ChangeDetectorRef }, { type: FocusTrapFactory }, { type: ViewContainerRef }, { type: OverlayKeyboardDispatcher }, { type: Directionality, decorators: [{ type: Optional }] } ]; NzDrawerComponent.propDecorators = { nzContent: [{ type: Input }], nzCloseIcon: [{ type: Input }], nzClosable: [{ type: Input }], nzMaskClosable: [{ type: Input }], nzMask: [{ type: Input }], nzCloseOnNavigation: [{ type: Input }], nzNoAnimation: [{ type: Input }], nzKeyboard: [{ type: Input }], nzTitle: [{ type: Input }], nzFooter: [{ type: Input }], nzPlacement: [{ type: Input }], nzMaskStyle: [{ type: Input }], nzBodyStyle: [{ type: Input }], nzWrapClassName: [{ type: Input }], nzWidth: [{ type: Input }], nzHeight: [{ type: Input }], nzZIndex: [{ type: Input }], nzOffsetX: [{ type: Input }], nzOffsetY: [{ type: Input }], nzVisible: [{ type: Input }], nzOnViewInit: [{ type: Output }], nzOnClose: [{ type: Output }], nzVisibleChange: [{ type: Output }], drawerTemplate: [{ type: ViewChild, args: ['drawerTemplate', { static: true },] }], bodyPortalOutlet: [{ type: ViewChild, args: [CdkPortalOutlet, { static: false },] }], contentFromContentChild: [{ type: ContentChild, args: [NzDrawerContentDirective, { static: true, read: TemplateRef },] }] }; __decorate([ InputBoolean(), __metadata("design:type", Boolean) ], NzDrawerComponent.prototype, "nzClosable", void 0); __decorate([ WithConfig(), InputBoolean(), __metadata("design:type", Boolean) ], NzDrawerComponent.prototype, "nzMaskClosable", void 0); __decorate([ WithConfig(), InputBoolean(), __metadata("design:type", Boolean) ], NzDrawerComponent.prototype, "nzMask", void 0); __decorate([ WithConfig(), InputBoolean(), __metadata("design:type", Boolean) ], NzDrawerComponent.prototype, "nzCloseOnNavigation", void 0); __decorate([ InputBoolean(), __metadata("design:type", Object) ], NzDrawerComponent.prototype, "nzNoAnimation", void 0); __decorate([ InputBoolean(), __metadata("design:type", Boolean) ], NzDrawerComponent.prototype, "nzKeyboard", void 0); __decorate([ WithConfig(), __metadata("design:type", String) ], NzDrawerComponent.prototype, "nzDirection", void 0); /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ class NzDrawerServiceModule { } NzDrawerServiceModule.decorators = [ { type: NgModule } ]; /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ class NzDrawerModule { } NzDrawerModule.decorators = [ { type: NgModule, args: [{ imports: [ BidiModule, CommonModule, OverlayModule, PortalModule, NzIconModule, NzOutletModule, NzNoAnimationModule, NzDrawerServiceModule ], exports: [NzDrawerComponent, NzDrawerContentDirective], declarations: [NzDrawerComponent, NzDrawerContentDirective], entryComponents: [NzDrawerComponent] },] } ]; class DrawerBuilderForService { constructor(overlay, options) { this.overlay = overlay; this.options = options; this.unsubscribe$ = new Subject(); /** pick {@link NzDrawerOptions.nzOnCancel} and omit this option */ const _a = this.options, { nzOnCancel } = _a, componentOption = __rest(_a, ["nzOnCancel"]); this.overlayRef = this.overlay.create(); this.drawerRef = this.overlayRef.attach(new ComponentPortal(NzDrawerComponent)).instance; this.updateOptions(componentOption); // Prevent repeatedly open drawer when tap focus element. this.drawerRef.savePreviouslyFocusedElement(); this.drawerRef.nzOnViewInit.pipe(takeUntil(this.unsubscribe$)).subscribe(() => { this.drawerRef.open(); }); this.drawerRef.nzOnClose.subscribe(() => { if (nzOnCancel) { nzOnCancel().then(canClose => { if (canClose !== false) { this.drawerRef.close(); } }); } else { this.drawerRef.close(); } }); this.drawerRef.afterClose.pipe(takeUntil(this.unsubscribe$)).subscribe(() => { this.overlayRef.dispose(); this.drawerRef = null; this.unsubscribe$.next(); this.unsubscribe$.complete(); }); } getInstance() { return this.drawerRef; } updateOptions(options) { Object.assign(this.drawerRef, options); } } class NzDrawerService { constructor(overlay) { this.overlay = overlay; } create(options) { return new DrawerBuilderForService(this.overlay, options).getInstance(); } } NzDrawerService.ɵprov = ɵɵdefineInjectable({ factory: function NzDrawerService_Factory() { return new NzDrawerService(ɵɵinject(Overlay)); }, token: NzDrawerService, providedIn: NzDrawerServiceModule }); NzDrawerService.decorators = [ { type: Injectable, args: [{ providedIn: NzDrawerServiceModule },] } ]; NzDrawerService.ctorParameters = () => [ { type: Overlay } ]; /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ /** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ /** * Generated bundle index. Do not edit. */ export { DRAWER_ANIMATE_DURATION, DrawerBuilderForService, NzDrawerComponent, NzDrawerContentDirective, NzDrawerModule, NzDrawerRef, NzDrawerService, NzDrawerServiceModule }; //# sourceMappingURL=ng-zorro-antd-drawer.js.map