@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
JavaScript
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"]}