@spartacus/storefront
Version:
Spartacus Storefront is a package that you can include in your application, which allows you to add default storefront features.
76 lines • 13.6 kB
JavaScript
import { ChangeDetectionStrategy, Component, } from '@angular/core';
import { BehaviorSubject } from 'rxjs';
import * as i0 from "@angular/core";
import * as i1 from "../current-product.service";
import * as i2 from "@spartacus/core";
import * as i3 from "../../../shared/components/star-rating/star-rating.component";
import * as i4 from "@angular/common";
export class ProductIntroComponent {
constructor(currentProductService, translationService, winRef) {
this.currentProductService = currentProductService;
this.translationService = translationService;
this.winRef = winRef;
this.reviewsTabAvailable = new BehaviorSubject(false);
this.product$ = this.currentProductService.getProduct();
}
ngAfterContentChecked() {
this.reviewsTabAvailable.next(!!this.getReviewsComponent());
}
// Scroll to views component on page and click "Reviews" tab
showReviews() {
// Use translated label for Reviews tab reference
this.translationService
.translate('TabPanelContainer.tabs.ProductReviewsTabComponent')
.subscribe((reviewsTabLabel) => {
const tabsComponent = this.getTabsComponent();
const reviewsTab = this.getTabByLabel(reviewsTabLabel, tabsComponent);
const reviewsComponent = this.getReviewsComponent();
if (reviewsTab && reviewsComponent) {
this.clickTabIfInactive(reviewsTab);
setTimeout(() => reviewsComponent.scrollIntoView({ behavior: 'smooth' }), 0);
}
})
.unsubscribe();
}
// NOTE: Does not currently exists as its own component
// but part of tabs component. This is likely to change in refactor.
getReviewsComponent() {
return this.winRef.document.querySelector('cx-product-reviews');
}
// Get Tabs Component if exists on page
getTabsComponent() {
return this.winRef.document.querySelector('cx-tab-paragraph-container');
}
// Click to activate tab if not already active
clickTabIfInactive(tab) {
if (!tab.classList.contains('active') ||
tab.classList.contains('toggled')) {
tab.click();
}
}
// Get Tab by label if exists on page
getTabByLabel(label, tabsComponent) {
if (tabsComponent) {
// NOTE: Reads through button tags to click on correct tab
// There may be a better way of doing this now/after refactor
const tabElements = tabsComponent.getElementsByTagName('button');
// Look through button tab elements until finding tab with label
for (const buttonElement of Array.from(tabElements)) {
if (buttonElement.innerHTML.includes(label)) {
return buttonElement;
}
}
}
}
}
ProductIntroComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ProductIntroComponent, deps: [{ token: i1.CurrentProductService }, { token: i2.TranslationService }, { token: i2.WindowRef }], target: i0.ɵɵFactoryTarget.Component });
ProductIntroComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: ProductIntroComponent, selector: "cx-product-intro", ngImport: i0, template: "<ng-container *ngIf=\"product$ | async as product\">\n <div class=\"rating\" *ngIf=\"product.averageRating\">\n <cx-star-rating [rating]=\"product?.averageRating\"></cx-star-rating>\n\n <div class=\"count\">({{ product?.numberOfReviews }})</div>\n\n <button\n *ngIf=\"reviewsTabAvailable | async\"\n class=\"btn btn-link cx-action-link\"\n (click)=\"showReviews()\"\n [attr.aria-label]=\"\n 'productSummary.showReviewsDetailed'\n | cxTranslate\n : {\n rating: product.averageRating | number: '1.0-1',\n count: product?.numberOfReviews\n }\n \"\n >\n {{ 'productSummary.showReviews' | cxTranslate }}\n </button>\n </div>\n <div class=\"rating\" *ngIf=\"!product.averageRating\">\n {{ 'productDetails.noReviews' | cxTranslate }}\n </div>\n <div class=\"code\">\n {{ 'productSummary.id' | cxTranslate }} {{ product?.code }}\n </div>\n</ng-container>\n", components: [{ type: i3.StarRatingComponent, selector: "cx-star-rating", inputs: ["disabled", "rating"], outputs: ["change"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], pipes: { "async": i4.AsyncPipe, "cxTranslate": i2.TranslatePipe, "number": i4.DecimalPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ProductIntroComponent, decorators: [{
type: Component,
args: [{
selector: 'cx-product-intro',
templateUrl: './product-intro.component.html',
changeDetection: ChangeDetectionStrategy.OnPush,
}]
}], ctorParameters: function () { return [{ type: i1.CurrentProductService }, { type: i2.TranslationService }, { type: i2.WindowRef }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-intro.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/cms-components/product/product-intro/product-intro.component.ts","../../../../../../projects/storefrontlib/cms-components/product/product-intro/product-intro.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,uBAAuB,EACvB,SAAS,GACV,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;;;;;;AAQnD,MAAM,OAAO,qBAAqB;IAKhC,YACY,qBAA4C,EAC9C,kBAAsC,EACpC,MAAiB;QAFjB,0BAAqB,GAArB,qBAAqB,CAAuB;QAC9C,uBAAkB,GAAlB,kBAAkB,CAAoB;QACpC,WAAM,GAAN,MAAM,CAAW;QAP7B,wBAAmB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAE1D,aAAQ,GAAwB,IAAI,CAAC,qBAAqB,CAAC,UAAU,EAAE,CAAC;IAMrE,CAAC;IAEJ,qBAAqB;QACnB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC;IAC9D,CAAC;IAED,4DAA4D;IAC5D,WAAW;QACT,iDAAiD;QACjD,IAAI,CAAC,kBAAkB;aACpB,SAAS,CAAC,mDAAmD,CAAC;aAC9D,SAAS,CAAC,CAAC,eAAe,EAAE,EAAE;YAC7B,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9C,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,eAAe,EAAE,aAAa,CAAC,CAAC;YACtE,MAAM,gBAAgB,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;YACpD,IAAI,UAAU,IAAI,gBAAgB,EAAE;gBAClC,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;gBACpC,UAAU,CACR,GAAG,EAAE,CAAC,gBAAgB,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,EAC7D,CAAC,CACF,CAAC;aACH;QACH,CAAC,CAAC;aACD,WAAW,EAAE,CAAC;IACnB,CAAC;IAED,uDAAuD;IACvD,oEAAoE;IAC5D,mBAAmB;QACzB,OAAO,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,oBAAoB,CAAC,CAAC;IAClE,CAAC;IAED,uCAAuC;IAC/B,gBAAgB;QACtB,OAAO,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,4BAA4B,CAAC,CAAC;IAC1E,CAAC;IAED,8CAA8C;IACtC,kBAAkB,CAAC,GAAgB;QACzC,IACE,CAAC,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC;YACjC,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,EACjC;YACA,GAAG,CAAC,KAAK,EAAE,CAAC;SACb;IACH,CAAC;IAED,qCAAqC;IAC7B,aAAa,CAAC,KAAa,EAAE,aAAsB;QACzD,IAAI,aAAa,EAAE;YACjB,0DAA0D;YAC1D,6DAA6D;YAC7D,MAAM,WAAW,GACf,aAAa,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;YAE/C,gEAAgE;YAChE,KAAK,MAAM,aAAa,IAAI,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE;gBACnD,IAAI,aAAa,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;oBAC3C,OAAO,aAAa,CAAC;iBACtB;aACF;SACF;IACH,CAAC;;kHAvEU,qBAAqB;sGAArB,qBAAqB,wDCdlC,i9BA6BA;2FDfa,qBAAqB;kBALjC,SAAS;mBAAC;oBACT,QAAQ,EAAE,kBAAkB;oBAC5B,WAAW,EAAE,gCAAgC;oBAC7C,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD","sourcesContent":["import {\n  AfterContentChecked,\n  ChangeDetectionStrategy,\n  Component,\n} from '@angular/core';\nimport { Product, TranslationService, WindowRef } from '@spartacus/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { CurrentProductService } from '../current-product.service';\n\n@Component({\n  selector: 'cx-product-intro',\n  templateUrl: './product-intro.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ProductIntroComponent implements AfterContentChecked {\n  reviewsTabAvailable = new BehaviorSubject<boolean>(false);\n\n  product$: Observable<Product> = this.currentProductService.getProduct();\n\n  constructor(\n    protected currentProductService: CurrentProductService,\n    private translationService: TranslationService,\n    protected winRef: WindowRef\n  ) {}\n\n  ngAfterContentChecked() {\n    this.reviewsTabAvailable.next(!!this.getReviewsComponent());\n  }\n\n  // Scroll to views component on page and click \"Reviews\" tab\n  showReviews() {\n    // Use translated label for Reviews tab reference\n    this.translationService\n      .translate('TabPanelContainer.tabs.ProductReviewsTabComponent')\n      .subscribe((reviewsTabLabel) => {\n        const tabsComponent = this.getTabsComponent();\n        const reviewsTab = this.getTabByLabel(reviewsTabLabel, tabsComponent);\n        const reviewsComponent = this.getReviewsComponent();\n        if (reviewsTab && reviewsComponent) {\n          this.clickTabIfInactive(reviewsTab);\n          setTimeout(\n            () => reviewsComponent.scrollIntoView({ behavior: 'smooth' }),\n            0\n          );\n        }\n      })\n      .unsubscribe();\n  }\n\n  // NOTE: Does not currently exists as its own component\n  // but part of tabs component. This is likely to change in refactor.\n  private getReviewsComponent(): Element {\n    return this.winRef.document.querySelector('cx-product-reviews');\n  }\n\n  // Get Tabs Component if exists on page\n  private getTabsComponent(): Element {\n    return this.winRef.document.querySelector('cx-tab-paragraph-container');\n  }\n\n  // Click to activate tab if not already active\n  private clickTabIfInactive(tab: HTMLElement): void {\n    if (\n      !tab.classList.contains('active') ||\n      tab.classList.contains('toggled')\n    ) {\n      tab.click();\n    }\n  }\n\n  // Get Tab by label if exists on page\n  private getTabByLabel(label: string, tabsComponent: Element): HTMLElement {\n    if (tabsComponent) {\n      // NOTE: Reads through button tags to click on correct tab\n      // There may be a better way of doing this now/after refactor\n      const tabElements: HTMLCollectionOf<HTMLElement> =\n        tabsComponent.getElementsByTagName('button');\n\n      // Look through button tab elements until finding tab with label\n      for (const buttonElement of Array.from(tabElements)) {\n        if (buttonElement.innerHTML.includes(label)) {\n          return buttonElement;\n        }\n      }\n    }\n  }\n}\n","<ng-container *ngIf=\"product$ | async as product\">\n  <div class=\"rating\" *ngIf=\"product.averageRating\">\n    <cx-star-rating [rating]=\"product?.averageRating\"></cx-star-rating>\n\n    <div class=\"count\">({{ product?.numberOfReviews }})</div>\n\n    <button\n      *ngIf=\"reviewsTabAvailable | async\"\n      class=\"btn btn-link cx-action-link\"\n      (click)=\"showReviews()\"\n      [attr.aria-label]=\"\n        'productSummary.showReviewsDetailed'\n          | cxTranslate\n            : {\n                rating: product.averageRating | number: '1.0-1',\n                count: product?.numberOfReviews\n              }\n      \"\n    >\n      {{ 'productSummary.showReviews' | cxTranslate }}\n    </button>\n  </div>\n  <div class=\"rating\" *ngIf=\"!product.averageRating\">\n    {{ 'productDetails.noReviews' | cxTranslate }}\n  </div>\n  <div class=\"code\">\n    {{ 'productSummary.id' | cxTranslate }} {{ product?.code }}\n  </div>\n</ng-container>\n"]}