UNPKG

@spartacus/storefront

Version:

Spartacus Storefront is a package that you can include in your application, which allows you to add default storefront features.

88 lines 15.1 kB
import { ChangeDetectorRef, Directive, EventEmitter, Input, Optional, Output, } from '@angular/core'; import { finalize, tap } from 'rxjs/operators'; import { ComponentCreateEvent, ComponentDestroyEvent, } from './events/component.event'; import * as i0 from "@angular/core"; import * as i1 from "../../services/cms-components.service"; import * as i2 from "@spartacus/core"; import * as i3 from "./services/component-handler.service"; import * as i4 from "./services/cms-injector.service"; /** * Directive used to facilitate instantiation of CMS driven dynamic components */ export class ComponentWrapperDirective { constructor(vcr, cmsComponentsService, injector, dynamicAttributeService, renderer, componentHandler, cmsInjector, eventService) { this.vcr = vcr; this.cmsComponentsService = cmsComponentsService; this.injector = injector; this.dynamicAttributeService = dynamicAttributeService; this.renderer = renderer; this.componentHandler = componentHandler; this.cmsInjector = cmsInjector; this.eventService = eventService; this.cxComponentRef = new EventEmitter(); } ngOnInit() { this.cmsComponentsService .determineMappings([this.cxComponentWrapper.flexType]) .subscribe(() => { if (this.cmsComponentsService.shouldRender(this.cxComponentWrapper.flexType)) { this.launchComponent(); } }); } launchComponent() { const componentMapping = this.cmsComponentsService.getMapping(this.cxComponentWrapper.flexType); if (!componentMapping) { return; } this.launcherResource = this.componentHandler .getLauncher(componentMapping, this.vcr, this.cmsInjector.getInjector(this.cxComponentWrapper.flexType, this.cxComponentWrapper.uid, this.injector), this.cmsComponentsService.getModule(this.cxComponentWrapper.flexType)) .pipe(tap(({ elementRef, componentRef }) => { this.cmpRef = componentRef; this.cxComponentRef.emit(componentRef); this.dispatchEvent(ComponentCreateEvent, elementRef); this.decorate(elementRef); this.injector.get(ChangeDetectorRef).markForCheck(); }), finalize(() => this.dispatchEvent(ComponentDestroyEvent))) .subscribe(); } /** * Dispatch the component event. * * The event is dispatched during creation and removal of the component. */ dispatchEvent(event, elementRef) { var _a; const payload = { typeCode: this.cxComponentWrapper.typeCode, id: this.cxComponentWrapper.uid, }; if (event === ComponentCreateEvent) { payload.host = elementRef === null || elementRef === void 0 ? void 0 : elementRef.nativeElement; } (_a = this.eventService) === null || _a === void 0 ? void 0 : _a.dispatch(payload, event); } decorate(elementRef) { this.dynamicAttributeService.addAttributesToComponent(elementRef.nativeElement, this.renderer, this.cxComponentWrapper); } ngOnDestroy() { if (this.launcherResource) { this.launcherResource.unsubscribe(); } } } ComponentWrapperDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ComponentWrapperDirective, deps: [{ token: i0.ViewContainerRef }, { token: i1.CmsComponentsService }, { token: i0.Injector }, { token: i2.DynamicAttributeService }, { token: i0.Renderer2 }, { token: i3.ComponentHandlerService }, { token: i4.CmsInjectorService }, { token: i2.EventService, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); ComponentWrapperDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "12.0.5", type: ComponentWrapperDirective, selector: "[cxComponentWrapper]", inputs: { cxComponentWrapper: "cxComponentWrapper" }, outputs: { cxComponentRef: "cxComponentRef" }, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ComponentWrapperDirective, decorators: [{ type: Directive, args: [{ selector: '[cxComponentWrapper]', }] }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i1.CmsComponentsService }, { type: i0.Injector }, { type: i2.DynamicAttributeService }, { type: i0.Renderer2 }, { type: i3.ComponentHandlerService }, { type: i4.CmsInjectorService }, { type: i2.EventService, decorators: [{ type: Optional }] }]; }, propDecorators: { cxComponentWrapper: [{ type: Input }], cxComponentRef: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component-wrapper.directive.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/cms-structure/page/component/component-wrapper.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  ComponentRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  Injector,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  Renderer2,\n  Type,\n  ViewContainerRef,\n} from '@angular/core';\nimport {\n  ContentSlotComponentData,\n  DynamicAttributeService,\n  EventService,\n} from '@spartacus/core';\nimport { Subscription } from 'rxjs';\nimport { finalize, tap } from 'rxjs/operators';\nimport { CmsComponentsService } from '../../services/cms-components.service';\nimport {\n  ComponentCreateEvent,\n  ComponentDestroyEvent,\n  ComponentEvent,\n} from './events/component.event';\nimport { CmsInjectorService } from './services/cms-injector.service';\nimport { ComponentHandlerService } from './services/component-handler.service';\n\n/**\n * Directive used to facilitate instantiation of CMS driven dynamic components\n */\n@Directive({\n  selector: '[cxComponentWrapper]',\n})\nexport class ComponentWrapperDirective implements OnInit, OnDestroy {\n  @Input() cxComponentWrapper: ContentSlotComponentData;\n  @Output() cxComponentRef = new EventEmitter<ComponentRef<any>>();\n\n  /**\n   * @deprecated since 2.0\n   *\n   * This property in unsafe, i.e.\n   * - cmpRef can be set later because of lazy loading or deferred loading\n   * - cmpRef can be not set at all if for example, web components are used as cms components\n   */\n  cmpRef?: ComponentRef<any>;\n\n  private launcherResource?: Subscription;\n\n  /**\n   * @deprecated since version 3.3\n   * Use the following constructor instead:\n   * ```\n   * constructor( protected vcr: ViewContainerRef,\n   * protected cmsComponentsService: CmsComponentsService,\n   * protected injector: Injector,\n   * protected dynamicAttributeService: DynamicAttributeService,\n   * protected renderer: Renderer2,\n   * protected componentHandler: ComponentHandlerService,\n   * protected cmsInjector: CmsInjectorService,\n   * protected eventService: EventService) {}\n   * ```\n   */\n  constructor(\n    vcr: ViewContainerRef,\n    cmsComponentsService: CmsComponentsService,\n    injector: Injector,\n    dynamicAttributeService: DynamicAttributeService,\n    renderer: Renderer2,\n    componentHandler: ComponentHandlerService,\n    cmsInjector: CmsInjectorService\n  );\n  constructor(\n    protected vcr: ViewContainerRef,\n    protected cmsComponentsService: CmsComponentsService,\n    protected injector: Injector,\n    protected dynamicAttributeService: DynamicAttributeService,\n    protected renderer: Renderer2,\n    protected componentHandler: ComponentHandlerService,\n    protected cmsInjector: CmsInjectorService,\n    @Optional() protected eventService?: EventService\n  ) {}\n\n  ngOnInit() {\n    this.cmsComponentsService\n      .determineMappings([this.cxComponentWrapper.flexType])\n      .subscribe(() => {\n        if (\n          this.cmsComponentsService.shouldRender(\n            this.cxComponentWrapper.flexType\n          )\n        ) {\n          this.launchComponent();\n        }\n      });\n  }\n\n  private launchComponent() {\n    const componentMapping = this.cmsComponentsService.getMapping(\n      this.cxComponentWrapper.flexType\n    );\n\n    if (!componentMapping) {\n      return;\n    }\n\n    this.launcherResource = this.componentHandler\n      .getLauncher(\n        componentMapping,\n        this.vcr,\n        this.cmsInjector.getInjector(\n          this.cxComponentWrapper.flexType,\n          this.cxComponentWrapper.uid,\n          this.injector\n        ),\n        this.cmsComponentsService.getModule(this.cxComponentWrapper.flexType)\n      )\n      .pipe(\n        tap(({ elementRef, componentRef }) => {\n          this.cmpRef = componentRef;\n\n          this.cxComponentRef.emit(componentRef);\n\n          this.dispatchEvent(ComponentCreateEvent, elementRef);\n          this.decorate(elementRef);\n          this.injector.get(ChangeDetectorRef).markForCheck();\n        }),\n        finalize(() => this.dispatchEvent(ComponentDestroyEvent))\n      )\n      .subscribe();\n  }\n\n  /**\n   * Dispatch the component event.\n   *\n   * The event is dispatched during creation and removal of the component.\n   */\n  protected dispatchEvent(\n    event: Type<ComponentEvent>,\n    elementRef?: ElementRef\n  ) {\n    const payload = {\n      typeCode: this.cxComponentWrapper.typeCode,\n      id: this.cxComponentWrapper.uid,\n    } as ComponentEvent;\n    if (event === ComponentCreateEvent) {\n      (payload as ComponentCreateEvent).host = elementRef?.nativeElement;\n    }\n    this.eventService?.dispatch(payload, event);\n  }\n\n  private decorate(elementRef: ElementRef): void {\n    this.dynamicAttributeService.addAttributesToComponent(\n      elementRef.nativeElement,\n      this.renderer,\n      this.cxComponentWrapper\n    );\n  }\n\n  ngOnDestroy() {\n    if (this.launcherResource) {\n      this.launcherResource.unsubscribe();\n    }\n  }\n}\n"]}