UNPKG

@doku-dev/doku-fragment

Version:

A new Angular UI library that moving away from Bootstrap and built from scratch.

133 lines 19.6 kB
import { DOCUMENT } from '@angular/common'; import { Inject, Injectable, } from '@angular/core'; import { NavigationStart } from '@angular/router'; import { ReplaySubject, filter, takeUntil } from 'rxjs'; import { DokuActiveModal, DokuModalRef } from './modal-ref'; import { ViewElement } from './view-element'; import * as i0 from "@angular/core"; import * as i1 from "../backdrop/backdrop.service"; import * as i2 from "@angular/router"; export class DokuModalService { constructor(envInjector, appRef, document, backdropService, injector, router) { this.envInjector = envInjector; this.appRef = appRef; this.document = document; this.backdropService = backdropService; this.injector = injector; this.router = router; this.modalRefs = []; this.destroy$ = new ReplaySubject(); this.router.events .pipe(filter((ev) => ev instanceof NavigationStart), takeUntil(this.destroy$)) .subscribe(() => this.closeAll()); } ngOnDestroy() { this.closeAll(); this.destroy$.next(true); this.destroy$.complete(); } open(componentOrTemplateRef, config) { const activeModal = new DokuActiveModal(); const view = ViewElement.createElement(componentOrTemplateRef, { environmentInjector: this.envInjector, applicationRef: this.appRef, document: this.document, activeModal: activeModal, injector: this.injector, }); // Add portal class (if exists) to the portal element if (config?.portalClass) { view.element.portal.classList.add(config.portalClass); } const componentInstance = view.content.componentRef?.instance || view.content.templateRef?.context; const modalRef = new DokuModalRef(this, componentInstance, view); const normalizedConfig = this.setViewByConfig(view, config); activeModal['_config'] = normalizedConfig; activeModal.close = modalRef.close.bind(modalRef); setTimeout(() => { const { backdropRef } = ViewElement.appendToBody(view, { backdropService: this.backdropService, document: this.document, }); modalRef['backdropRef'] = backdropRef; // Add overflow-hidden class to the document body this.document.body.classList.add('d-body-overflow-hidden'); view.content.componentRef?.changeDetectorRef.detectChanges(); view.content.templateRef?.detectChanges(); // Remove focus on trigger element this.document.activeElement?.blur(); }, 0); this.modalRefs.push(modalRef); return modalRef; } closeAll() { this.modalRefs.forEach((ref) => this.close(ref)); } close(modalRef) { if (!modalRef['view']) return; ViewElement.removeFromBody(modalRef['view'], { backdropService: this.backdropService, document: this.document, backdropRef: modalRef['backdropRef'], }); modalRef['view'] = undefined; this.modalRefs = this.modalRefs.filter((ref) => ref !== modalRef); // Remove overflow-hidden class from document body if all of the modals have been closed. if (!this.modalRefs.length) { this.document.body.classList.remove('d-body-overflow-hidden'); } } /** * @returns Normalized DokuModalConfig */ setViewByConfig(view, config) { const normalizedConfig = { ...config }; if (config?.centered === undefined || config.centered === null) { normalizedConfig.centered = true; } if (!config?.variant) normalizedConfig.variant = 'info'; if (!config?.size) normalizedConfig.size = 'medium'; if (normalizedConfig.variant) { view.element.modal.classList.add(`d-modal-variant-${normalizedConfig.variant}`); if (normalizedConfig.variant === 'form') { view.element.modal.classList.add('d-modal-scrollable-content'); } if (normalizedConfig.variant === 'confirmation') { normalizedConfig.confirmationOptions = { ...normalizedConfig.confirmationOptions }; // Set default size if not set explicitly. if (!config?.size) normalizedConfig.size = 'small'; // Set default icon type if not set explicitly. if (!config?.confirmationOptions?.iconType) { normalizedConfig.confirmationOptions.iconType = 'warning'; } // Set default show icon if not set explicitly. if (!config?.confirmationOptions?.showIcon) { normalizedConfig.confirmationOptions.showIcon = true; } } } if (normalizedConfig.centered) { view.element.modal.classList.add('d-modal-centered'); } if (normalizedConfig.size) { view.element.modal.classList.add(`d-modal-${normalizedConfig.size}`); } return normalizedConfig; } } DokuModalService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuModalService, deps: [{ token: i0.EnvironmentInjector }, { token: i0.ApplicationRef }, { token: DOCUMENT }, { token: i1.DokuBackdropService }, { token: i0.Injector }, { token: i2.Router }], target: i0.ɵɵFactoryTarget.Injectable }); DokuModalService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuModalService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuModalService, decorators: [{ type: Injectable, args: [{ providedIn: 'root', }] }], ctorParameters: function () { return [{ type: i0.EnvironmentInjector }, { type: i0.ApplicationRef }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: i1.DokuBackdropService }, { type: i0.Injector }, { type: i2.Router }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.service.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/modal/modal.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  ApplicationRef,\n  EnvironmentInjector,\n  Inject,\n  Injectable,\n  Injector,\n  OnDestroy,\n  TemplateRef,\n} from '@angular/core';\nimport { NavigationStart, Router } from '@angular/router';\nimport { ReplaySubject, filter, takeUntil } from 'rxjs';\nimport { DokuBackdropService } from '../backdrop/backdrop.service';\nimport { DokuActiveModal, DokuModalRef } from './modal-ref';\nimport { ComponentType, DokuModalConfig } from './modal.interface';\nimport { ViewElement } from './view-element';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class DokuModalService implements OnDestroy {\n  protected modalRefs: DokuModalRef<unknown>[] = [];\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    private envInjector: EnvironmentInjector,\n    private appRef: ApplicationRef,\n    @Inject(DOCUMENT) private document: Document,\n    private backdropService: DokuBackdropService,\n    private injector: Injector,\n    private router: Router\n  ) {\n    this.router.events\n      .pipe(\n        filter((ev) => ev instanceof NavigationStart),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(() => this.closeAll());\n  }\n\n  ngOnDestroy(): void {\n    this.closeAll();\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  open<T>(\n    componentOrTemplateRef: ComponentType<T> | TemplateRef<T>,\n    config?: DokuModalConfig\n  ): DokuModalRef<T> {\n    const activeModal = new DokuActiveModal();\n\n    const view = ViewElement.createElement(componentOrTemplateRef, {\n      environmentInjector: this.envInjector,\n      applicationRef: this.appRef,\n      document: this.document,\n      activeModal: activeModal,\n      injector: this.injector,\n    });\n\n    // Add portal class (if exists) to the portal element\n    if (config?.portalClass) {\n      view.element.portal.classList.add(config.portalClass);\n    }\n\n    const componentInstance =\n      view.content.componentRef?.instance || view.content.templateRef?.context;\n\n    const modalRef = new DokuModalRef<T>(this, <T>componentInstance, view);\n\n    const normalizedConfig = this.setViewByConfig(view, config);\n\n    activeModal['_config'] = normalizedConfig;\n    activeModal.close = modalRef.close.bind(modalRef);\n\n    setTimeout(() => {\n      const { backdropRef } = ViewElement.appendToBody(view, {\n        backdropService: this.backdropService,\n        document: this.document,\n      });\n\n      modalRef['backdropRef'] = backdropRef;\n\n      // Add overflow-hidden class to the document body\n      this.document.body.classList.add('d-body-overflow-hidden');\n\n      view.content.componentRef?.changeDetectorRef.detectChanges();\n      view.content.templateRef?.detectChanges();\n\n      // Remove focus on trigger element\n      (this.document.activeElement as HTMLElement)?.blur();\n    }, 0);\n\n    this.modalRefs.push(modalRef);\n    return modalRef;\n  }\n\n  closeAll() {\n    this.modalRefs.forEach((ref) => this.close(ref));\n  }\n\n  protected close(modalRef: DokuModalRef<unknown>) {\n    if (!modalRef['view']) return;\n    ViewElement.removeFromBody(modalRef['view'], {\n      backdropService: this.backdropService,\n      document: this.document,\n      backdropRef: modalRef['backdropRef'],\n    });\n    modalRef['view'] = undefined;\n    this.modalRefs = this.modalRefs.filter((ref) => ref !== modalRef);\n\n    // Remove overflow-hidden class from document body if all of the modals have been closed.\n    if (!this.modalRefs.length) {\n      this.document.body.classList.remove('d-body-overflow-hidden');\n    }\n  }\n\n  /**\n   * @returns Normalized DokuModalConfig\n   */\n  private setViewByConfig(\n    view: ReturnType<typeof ViewElement.createElement>,\n    config?: DokuModalConfig\n  ): DokuModalConfig {\n    const normalizedConfig = { ...config };\n\n    if (config?.centered === undefined || config.centered === null) {\n      normalizedConfig.centered = true;\n    }\n    if (!config?.variant) normalizedConfig.variant = 'info';\n    if (!config?.size) normalizedConfig.size = 'medium';\n\n    if (normalizedConfig.variant) {\n      view.element.modal.classList.add(`d-modal-variant-${normalizedConfig.variant}`);\n\n      if (normalizedConfig.variant === 'form') {\n        view.element.modal.classList.add('d-modal-scrollable-content');\n      }\n\n      if (normalizedConfig.variant === 'confirmation') {\n        normalizedConfig.confirmationOptions = { ...normalizedConfig.confirmationOptions };\n\n        // Set default size if not set explicitly.\n        if (!config?.size) normalizedConfig.size = 'small';\n        // Set default icon type if not set explicitly.\n        if (!config?.confirmationOptions?.iconType) {\n          normalizedConfig.confirmationOptions.iconType = 'warning';\n        }\n        // Set default show icon if not set explicitly.\n        if (!config?.confirmationOptions?.showIcon) {\n          normalizedConfig.confirmationOptions.showIcon = true;\n        }\n      }\n    }\n\n    if (normalizedConfig.centered) {\n      view.element.modal.classList.add('d-modal-centered');\n    }\n\n    if (normalizedConfig.size) {\n      view.element.modal.classList.add(`d-modal-${normalizedConfig.size}`);\n    }\n\n    return normalizedConfig;\n  }\n}\n"]}