@spartacus/storefront
Version:
Spartacus Storefront is a package that you can include in your application, which allows you to add default storefront features.
96 lines • 17.9 kB
JavaScript
import { ChangeDetectionStrategy, Component, ViewChildren, } from '@angular/core';
import { combineLatest } from 'rxjs';
import { distinctUntilChanged, map, switchMap, take } from 'rxjs/operators';
import { ComponentWrapperDirective } from '../../../cms-structure/page/component/component-wrapper.directive';
import { BREAKPOINT } from '../../../layout/config/layout-config';
import * as i0 from "@angular/core";
import * as i1 from "../../../cms-structure/page/model/index";
import * as i2 from "@spartacus/core";
import * as i3 from "../../../layout/breakpoint/breakpoint.service";
import * as i4 from "@angular/common";
import * as i5 from "../../../cms-structure/outlet/outlet.directive";
import * as i6 from "../../../cms-structure/page/component/component-wrapper.directive";
export class TabParagraphContainerComponent {
constructor(componentData, cmsService, winRef, breakpointService) {
this.componentData = componentData;
this.cmsService = cmsService;
this.winRef = winRef;
this.breakpointService = breakpointService;
this.activeTabNum = 0;
this.tabTitleParams = [];
this.components$ = this.componentData.data$.pipe(distinctUntilChanged((x, y) => (x === null || x === void 0 ? void 0 : x.components) === (y === null || y === void 0 ? void 0 : y.components)), switchMap((data) => {
var _a;
return combineLatest(((_a = data === null || data === void 0 ? void 0 : data.components) !== null && _a !== void 0 ? _a : '').split(' ').map((component) => this.cmsService.getComponentData(component).pipe(distinctUntilChanged(), map((tab) => {
if (!tab) {
return undefined;
}
if (!tab.flexType) {
tab = Object.assign(Object.assign({}, tab), { flexType: tab.typeCode });
}
return Object.assign(Object.assign({}, tab), { title: `${data.uid}.tabs.${tab.uid}` });
}))));
}));
}
select(tabNum, event) {
var _a;
(_a = this.breakpointService) === null || _a === void 0 ? void 0 : _a.isDown(BREAKPOINT.sm).pipe(take(1)).subscribe((res) => {
var _a, _b;
if (res) {
this.activeTabNum = this.activeTabNum === tabNum ? -1 : tabNum;
if (event && (event === null || event === void 0 ? void 0 : event.target)) {
const target = event.target;
const parentNode = target.parentNode;
(_b = (_a = this.winRef) === null || _a === void 0 ? void 0 : _a.nativeWindow) === null || _b === void 0 ? void 0 : _b.scrollTo(0, parentNode.offsetTop);
}
}
else {
this.activeTabNum = tabNum;
}
});
}
ngOnInit() {
var _a, _b, _c, _d, _e;
this.activeTabNum =
(_e = (_d = (_c = (_b = (_a = this.winRef) === null || _a === void 0 ? void 0 : _a.nativeWindow) === null || _b === void 0 ? void 0 : _b.history) === null || _c === void 0 ? void 0 : _c.state) === null || _d === void 0 ? void 0 : _d.activeTab) !== null && _e !== void 0 ? _e : this.activeTabNum;
}
ngAfterViewInit() {
// If the sub cms components data exist, the components created before ngAfterViewInit are called.
// In this case, the title parameters are directly pulled from them.
if (this.children.length > 0) {
this.getTitleParams(this.children);
}
}
tabCompLoaded(componentRef) {
this.tabTitleParams.push(componentRef.instance.tabTitleParam$);
}
getTitleParams(children) {
children.forEach((comp) => {
var _a;
if ((_a = comp.cmpRef) === null || _a === void 0 ? void 0 : _a.instance.tabTitleParam$) {
this.tabTitleParams.push(comp.cmpRef.instance.tabTitleParam$);
}
else {
this.tabTitleParams.push(null);
}
});
}
ngOnDestroy() {
if (this.subscription) {
this.subscription.unsubscribe();
}
}
}
TabParagraphContainerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: TabParagraphContainerComponent, deps: [{ token: i1.CmsComponentData }, { token: i2.CmsService }, { token: i2.WindowRef }, { token: i3.BreakpointService }], target: i0.ɵɵFactoryTarget.Component });
TabParagraphContainerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: TabParagraphContainerComponent, selector: "cx-tab-paragraph-container", viewQueries: [{ propertyName: "children", predicate: ComponentWrapperDirective, descendants: true }], ngImport: i0, template: "<ng-container *ngIf=\"components$ | async as components\">\n <ng-container *ngFor=\"let component of components; let i = index\">\n <ng-container *ngIf=\"component\">\n <button [class.active]=\"i === activeTabNum\" (click)=\"select(i, $event)\">\n {{\n component.title | cxTranslate: { param: tabTitleParams[i] | async }\n }}\n </button>\n </ng-container>\n </ng-container>\n\n <ng-container *ngFor=\"let component of components; let i = index\">\n <ng-container *ngIf=\"component\">\n <div [class.active]=\"i === activeTabNum\">\n <ng-template [cxOutlet]=\"component.flexType\" [cxOutletContext]=\"{}\">\n <ng-container\n [cxComponentWrapper]=\"component\"\n (cxComponentRef)=\"tabCompLoaded($event)\"\n ></ng-container>\n </ng-template>\n </div>\n </ng-container>\n </ng-container>\n</ng-container>\n", directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.OutletDirective, selector: "[cxOutlet]", inputs: ["cxOutlet", "cxOutletContext", "cxOutletDefer"], outputs: ["loaded"] }, { type: i6.ComponentWrapperDirective, selector: "[cxComponentWrapper]", inputs: ["cxComponentWrapper"], outputs: ["cxComponentRef"] }], pipes: { "async": i4.AsyncPipe, "cxTranslate": i2.TranslatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: TabParagraphContainerComponent, decorators: [{
type: Component,
args: [{
selector: 'cx-tab-paragraph-container',
templateUrl: './tab-paragraph-container.component.html',
changeDetection: ChangeDetectionStrategy.OnPush,
}]
}], ctorParameters: function () { return [{ type: i1.CmsComponentData }, { type: i2.CmsService }, { type: i2.WindowRef }, { type: i3.BreakpointService }]; }, propDecorators: { children: [{
type: ViewChildren,
args: [ComponentWrapperDirective]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-paragraph-container.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/cms-components/content/tab-paragraph-container/tab-paragraph-container.component.ts","../../../../../../projects/storefrontlib/cms-components/content/tab-paragraph-container/tab-paragraph-container.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  OnDestroy,\n  OnInit,\n  QueryList,\n  ViewChildren,\n} from '@angular/core';\nimport {\n  CmsService,\n  CMSTabParagraphContainer,\n  WindowRef,\n} from '@spartacus/core';\nimport { combineLatest, Observable, Subscription } from 'rxjs';\nimport { distinctUntilChanged, map, switchMap, take } from 'rxjs/operators';\nimport { ComponentWrapperDirective } from '../../../cms-structure/page/component/component-wrapper.directive';\nimport { CmsComponentData } from '../../../cms-structure/page/model/index';\nimport { BreakpointService } from '../../../layout/breakpoint/breakpoint.service';\nimport { BREAKPOINT } from '../../../layout/config/layout-config';\n\n@Component({\n  selector: 'cx-tab-paragraph-container',\n  templateUrl: './tab-paragraph-container.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TabParagraphContainerComponent\n  implements AfterViewInit, OnInit, OnDestroy\n{\n  activeTabNum = 0;\n\n  @ViewChildren(ComponentWrapperDirective)\n  children!: QueryList<ComponentWrapperDirective>;\n\n  tabTitleParams: (Observable<any> | null)[] = [];\n\n  // TODO: it is not used any more, so can be removed in 5.0\n  subscription: Subscription;\n\n  constructor(\n    public componentData: CmsComponentData<CMSTabParagraphContainer>,\n    protected cmsService: CmsService,\n    protected winRef: WindowRef,\n    protected breakpointService: BreakpointService\n  ) {}\n\n  components$: Observable<any[]> = this.componentData.data$.pipe(\n    distinctUntilChanged((x, y) => x?.components === y?.components),\n    switchMap((data) =>\n      combineLatest(\n        (data?.components ?? '').split(' ').map((component) =>\n          this.cmsService.getComponentData<any>(component).pipe(\n            distinctUntilChanged(),\n            map((tab) => {\n              if (!tab) {\n                return undefined;\n              }\n\n              if (!tab.flexType) {\n                tab = {\n                  ...tab,\n                  flexType: tab.typeCode,\n                };\n              }\n\n              return {\n                ...tab,\n                title: `${data.uid}.tabs.${tab.uid}`,\n              };\n            })\n          )\n        )\n      )\n    )\n  );\n\n  select(tabNum: number, event?: MouseEvent): void {\n    this.breakpointService\n      ?.isDown(BREAKPOINT.sm)\n      .pipe(take(1))\n      .subscribe((res) => {\n        if (res) {\n          this.activeTabNum = this.activeTabNum === tabNum ? -1 : tabNum;\n          if (event && event?.target) {\n            const target = event.target as HTMLElement;\n            const parentNode = target.parentNode as HTMLElement;\n            this.winRef?.nativeWindow?.scrollTo(0, parentNode.offsetTop);\n          }\n        } else {\n          this.activeTabNum = tabNum;\n        }\n      });\n  }\n\n  ngOnInit(): void {\n    this.activeTabNum =\n      this.winRef?.nativeWindow?.history?.state?.activeTab ?? this.activeTabNum;\n  }\n\n  ngAfterViewInit(): void {\n    // If the sub cms components data exist, the components created before ngAfterViewInit are called.\n    // In this case, the title parameters are directly pulled from them.\n    if (this.children.length > 0) {\n      this.getTitleParams(this.children);\n    }\n  }\n\n  tabCompLoaded(componentRef: any): void {\n    this.tabTitleParams.push(componentRef.instance.tabTitleParam$);\n  }\n\n  private getTitleParams(children: QueryList<ComponentWrapperDirective>) {\n    children.forEach((comp) => {\n      if (comp.cmpRef?.instance.tabTitleParam$) {\n        this.tabTitleParams.push(comp.cmpRef.instance.tabTitleParam$);\n      } else {\n        this.tabTitleParams.push(null);\n      }\n    });\n  }\n\n  ngOnDestroy(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n    }\n  }\n}\n","<ng-container *ngIf=\"components$ | async as components\">\n  <ng-container *ngFor=\"let component of components; let i = index\">\n    <ng-container *ngIf=\"component\">\n      <button [class.active]=\"i === activeTabNum\" (click)=\"select(i, $event)\">\n        {{\n          component.title | cxTranslate: { param: tabTitleParams[i] | async }\n        }}\n      </button>\n    </ng-container>\n  </ng-container>\n\n  <ng-container *ngFor=\"let component of components; let i = index\">\n    <ng-container *ngIf=\"component\">\n      <div [class.active]=\"i === activeTabNum\">\n        <ng-template [cxOutlet]=\"component.flexType\" [cxOutletContext]=\"{}\">\n          <ng-container\n            [cxComponentWrapper]=\"component\"\n            (cxComponentRef)=\"tabCompLoaded($event)\"\n          ></ng-container>\n        </ng-template>\n      </div>\n    </ng-container>\n  </ng-container>\n</ng-container>\n"]}