UNPKG

@doku-dev/doku-fragment

Version:

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

70 lines 12.5 kB
import { Injector, TemplateRef, createComponent, } from '@angular/core'; import { DokuActiveModal } from './modal-ref'; export class ViewElement { static createElement(content, props) { let contentTemplateRef; let contentComponentRef; const elementInjector = Injector.create({ providers: [{ provide: DokuActiveModal, useValue: props.activeModal }], parent: props.injector, }); if (content instanceof TemplateRef) { contentTemplateRef = content.createEmbeddedView({ $implicit: props.activeModal }, elementInjector); props.applicationRef.attachView(contentTemplateRef); } else { contentComponentRef = createComponent(content, { environmentInjector: props.environmentInjector, elementInjector, }); contentComponentRef.location.nativeElement.classList.add('d-modal-content-host'); props.applicationRef.attachView(contentComponentRef.hostView); } const modalContentElement = this.createModalContentElement({ document: props.document }); const modalElement = this.createModalElement({ document: props.document }); const portalElement = this.createPortalElement({ document: props.document }); return { element: { portal: portalElement, modal: modalElement, modalContent: modalContentElement }, content: { componentRef: contentComponentRef, templateRef: contentTemplateRef }, }; } static createModalElement(props) { const element = props.document.createElement('div'); element.className = 'd-modal'; return element; } static createModalContentElement(props) { const element = props.document.createElement('div'); element.className = 'd-modal-content'; return element; } static createPortalElement(props) { const element = props.document.createElement('div'); element.className = 'd-modal-portal'; return element; } static appendToBody(view, props) { const backdropRef = props.backdropService.open(); // Append content (from component or template) to modal content if (view.content.componentRef) { view.element.modalContent.appendChild(view.content.componentRef.location.nativeElement); } if (view.content.templateRef) { view.element.modalContent.append(...view.content.templateRef.rootNodes); } // Append modal content to modal view.element.modal.appendChild(view.element.modalContent); // Append modal to portal view.element.portal.appendChild(view.element.modal); // Append portal to document body props.document.body.appendChild(view.element.portal); return { backdropRef }; } static removeFromBody(view, props) { view.content.componentRef?.destroy(); view.content.templateRef?.destroy(); props.document.body.removeChild(view.element.portal); props.backdropRef?.close(); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"view-element.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/modal/view-element.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ApplicationRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  EnvironmentInjector,\n  Injector,\n  TemplateRef,\n  createComponent,\n} from '@angular/core';\nimport { DokuBackdropRef } from '../backdrop/backdrop-ref';\nimport { DokuBackdropService } from '../backdrop/backdrop.service';\nimport { DokuActiveModal } from './modal-ref';\nimport { ComponentType } from './modal.interface';\n\nexport class ViewElement {\n  static createElement<T>(\n    content: ComponentType<T> | TemplateRef<T>,\n    props: {\n      environmentInjector: EnvironmentInjector;\n      applicationRef: ApplicationRef;\n      injector: Injector;\n      document: Document;\n      activeModal: DokuActiveModal;\n    }\n  ) {\n    let contentTemplateRef: EmbeddedViewRef<T> | undefined;\n    let contentComponentRef: ComponentRef<T> | undefined;\n\n    const elementInjector = Injector.create({\n      providers: [{ provide: DokuActiveModal, useValue: props.activeModal }],\n      parent: props.injector,\n    });\n\n    if (content instanceof TemplateRef) {\n      contentTemplateRef = content.createEmbeddedView(\n        <T>{ $implicit: props.activeModal },\n        elementInjector\n      );\n      props.applicationRef.attachView(contentTemplateRef);\n    } else {\n      contentComponentRef = createComponent(content, {\n        environmentInjector: props.environmentInjector,\n        elementInjector,\n      });\n      (contentComponentRef.location.nativeElement as HTMLElement).classList.add(\n        'd-modal-content-host'\n      );\n      props.applicationRef.attachView(contentComponentRef.hostView);\n    }\n\n    const modalContentElement = this.createModalContentElement({ document: props.document });\n    const modalElement = this.createModalElement({ document: props.document });\n    const portalElement = this.createPortalElement({ document: props.document });\n\n    return {\n      element: { portal: portalElement, modal: modalElement, modalContent: modalContentElement },\n      content: { componentRef: contentComponentRef, templateRef: contentTemplateRef },\n    };\n  }\n\n  private static createModalElement(props: { document: Document }) {\n    const element: HTMLDivElement = props.document.createElement('div');\n    element.className = 'd-modal';\n    return element;\n  }\n\n  private static createModalContentElement(props: { document: Document }) {\n    const element: HTMLDivElement = props.document.createElement('div');\n    element.className = 'd-modal-content';\n    return element;\n  }\n\n  private static createPortalElement(props: { document: Document }) {\n    const element: HTMLDivElement = props.document.createElement('div');\n    element.className = 'd-modal-portal';\n    return element;\n  }\n\n  static appendToBody(\n    view: ReturnType<typeof this.createElement>,\n    props: { backdropService: DokuBackdropService; document: Document }\n  ) {\n    const backdropRef = props.backdropService.open();\n\n    // Append content (from component or template) to modal content\n    if (view.content.componentRef) {\n      view.element.modalContent.appendChild(view.content.componentRef.location.nativeElement);\n    }\n    if (view.content.templateRef) {\n      view.element.modalContent.append(...view.content.templateRef.rootNodes);\n    }\n\n    // Append modal content to modal\n    view.element.modal.appendChild(view.element.modalContent);\n\n    // Append modal to portal\n    view.element.portal.appendChild(view.element.modal);\n\n    // Append portal to document body\n    props.document.body.appendChild(view.element.portal);\n\n    return { backdropRef };\n  }\n\n  static removeFromBody(\n    view: ReturnType<typeof this.createElement>,\n    props: {\n      backdropService: DokuBackdropService;\n      document: Document;\n      backdropRef?: DokuBackdropRef;\n    }\n  ) {\n    view.content.componentRef?.destroy();\n    view.content.templateRef?.destroy();\n\n    props.document.body.removeChild(view.element.portal);\n    props.backdropRef?.close();\n  }\n}\n"]}