UNPKG

@spartacus/storefront

Version:

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

90 lines 16.6 kB
import { ChangeDetectionStrategy, Component, Input, isDevMode, } from '@angular/core'; import { tap } from 'rxjs/operators'; import { ICON_TYPE } from '../../../cms-components/misc/icon/icon.model'; import * as i0 from "@angular/core"; import * as i1 from "./carousel.service"; import * as i2 from "../../../cms-components/misc/icon/icon.component"; import * as i3 from "@angular/common"; import * as i4 from "@spartacus/core"; /** * Generic carousel component that can be used to render any carousel items, * such as products, images, banners, or any component. Carousel items are * rendered in so-called carousel slides, and the previous/next buttons as well as * the indicator-buttons can used to navigate the slides. * * The component uses an array of Observables (`items$`) as an input, to allow * for lazy loading of items. * * The number of items per slide is calculated with the `itemWidth`, which can given * in pixels or percentage. * * To allow for flexible rendering of items, the rendering is delegated to the * given `template`. This allows for maximum flexibility. */ export class CarouselComponent { constructor(el, service) { this.el = el; this.service = service; /** * Specifies the minimum size of the carousel item, either in px or %. * This value is used for the calculation of numbers per carousel, so that * the number of carousel items is dynamic. The calculation uses the `itemWidth` * and the host element `clientWidth`, so that the carousel is reusable in * different layouts (for example in a 50% grid). */ this.itemWidth = '300px'; /** * Indicates whether the visual indicators are used. */ this.hideIndicators = false; this.indicatorIcon = ICON_TYPE.CIRCLE; this.previousIcon = ICON_TYPE.CARET_LEFT; this.nextIcon = ICON_TYPE.CARET_RIGHT; } set setItems(inputItems) { this.items = inputItems; //Reset slider when changing products this.activeSlide = 0; } ngOnInit() { if (!this.template && isDevMode()) { console.error('No template reference provided to render the carousel items for the `cx-carousel`'); return; } this.size$ = this.service .getItemsPerSlide(this.el.nativeElement, this.itemWidth) .pipe(tap(() => (this.activeSlide = 0))); } getSlideNumber(size, currentIndex) { let normalizedCurrentIndex = currentIndex + 1; return Math.ceil(normalizedCurrentIndex / size); } } CarouselComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CarouselComponent, deps: [{ token: i0.ElementRef }, { token: i1.CarouselService }], target: i0.ɵɵFactoryTarget.Component }); CarouselComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: CarouselComponent, selector: "cx-carousel", inputs: { title: "title", setItems: ["items", "setItems"], template: "template", itemWidth: "itemWidth", hideIndicators: "hideIndicators", indicatorIcon: "indicatorIcon", previousIcon: "previousIcon", nextIcon: "nextIcon" }, ngImport: i0, template: "<ng-container *ngIf=\"items?.length > 0 && (size$ | async) as size\">\n <h2 *ngIf=\"title\">{{ title }}</h2>\n\n <div class=\"carousel-panel\" [ngClass]=\"'size-' + size\">\n <button\n *ngIf=\"size < items.length\"\n class=\"previous\"\n (click)=\"activeSlide = activeSlide - size\"\n [disabled]=\"activeSlide === 0\"\n [attr.aria-label]=\"'carousel.previousSlide' | cxTranslate\"\n >\n <cx-icon [type]=\"previousIcon\"></cx-icon>\n </button>\n\n <div class=\"slides\">\n <ng-container *ngFor=\"let _ of items; let i = index\">\n <div\n class=\"slide\"\n *ngIf=\"i % size === 0\"\n [class.active]=\"i === activeSlide\"\n >\n <ng-container\n *ngFor=\"let item of items | slice: i:i + size; let j = index\"\n >\n <div\n *ngIf=\"item | async as data\"\n class=\"item\"\n [class.active]=\"i === activeSlide\"\n >\n <ng-container\n *ngTemplateOutlet=\"template; context: { item: data }\"\n ></ng-container>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </div>\n\n <button\n *ngIf=\"size < items.length\"\n class=\"next\"\n (click)=\"activeSlide = activeSlide + size\"\n tabindex=\"-1\"\n [disabled]=\"activeSlide > items.length - size - 1\"\n [attr.aria-label]=\"'carousel.nextSlide' | cxTranslate\"\n >\n <cx-icon [type]=\"nextIcon\"></cx-icon>\n </button>\n </div>\n\n <div *ngIf=\"!hideIndicators && size < items.length\" class=\"indicators\">\n <ng-container *ngFor=\"let _ of items; let i = index\">\n <button\n *ngIf=\"i % size === 0\"\n (focus)=\"activeSlide = i\"\n [disabled]=\"i === activeSlide\"\n [attr.aria-label]=\"\n 'carousel.slideNumber'\n | cxTranslate: { currentSlideNumber: getSlideNumber(size, i) }\n \"\n tabindex=\"-1\"\n >\n <cx-icon [type]=\"indicatorIcon\" aria-hidden=\"true\"></cx-icon>\n </button>\n </ng-container>\n </div>\n</ng-container>\n", components: [{ type: i2.IconComponent, selector: "cx-icon,[cxIcon]", inputs: ["cxIcon", "type"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "async": i3.AsyncPipe, "cxTranslate": i4.TranslatePipe, "slice": i3.SlicePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CarouselComponent, decorators: [{ type: Component, args: [{ selector: 'cx-carousel', templateUrl: './carousel.component.html', changeDetection: ChangeDetectionStrategy.OnPush, }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.CarouselService }]; }, propDecorators: { title: [{ type: Input }], setItems: [{ type: Input, args: ['items'] }], template: [{ type: Input }], itemWidth: [{ type: Input }], hideIndicators: [{ type: Input }], indicatorIcon: [{ type: Input }], previousIcon: [{ type: Input }], nextIcon: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"carousel.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/shared/components/carousel/carousel.component.ts","../../../../../../projects/storefrontlib/shared/components/carousel/carousel.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EAET,KAAK,EACL,SAAS,GAGV,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,SAAS,EAAE,MAAM,8CAA8C,CAAC;;;;;;AAGzE;;;;;;;;;;;;;;GAcG;AAMH,MAAM,OAAO,iBAAiB;IA6C5B,YAAsB,EAAc,EAAY,OAAwB;QAAlD,OAAE,GAAF,EAAE,CAAY;QAAY,YAAO,GAAP,OAAO,CAAiB;QArBxE;;;;;;WAMG;QACM,cAAS,GAAG,OAAO,CAAC;QAE7B;;WAEG;QACM,mBAAc,GAAG,KAAK,CAAC;QAEvB,kBAAa,GAAG,SAAS,CAAC,MAAM,CAAC;QACjC,iBAAY,GAAG,SAAS,CAAC,UAAU,CAAC;QACpC,aAAQ,GAAG,SAAS,CAAC,WAAW,CAAC;IAKiC,CAAC;IAlC5E,IACI,QAAQ,CAAC,UAA6B;QACxC,IAAI,CAAC,KAAK,GAAG,UAAU,CAAC;QACxB,qCAAqC;QACrC,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;IACvB,CAAC;IA+BD,QAAQ;QACN,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,SAAS,EAAE,EAAE;YACjC,OAAO,CAAC,KAAK,CACX,mFAAmF,CACpF,CAAC;YACF,OAAO;SACR;QACD,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO;aACtB,gBAAgB,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,IAAI,CAAC,SAAS,CAAC;aACvD,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7C,CAAC;IAED,cAAc,CAAC,IAAY,EAAE,YAAoB;QAC/C,IAAI,sBAAsB,GAAG,YAAY,GAAG,CAAC,CAAC;QAC9C,OAAO,IAAI,CAAC,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,CAAC;IAClD,CAAC;;8GA9DU,iBAAiB;kGAAjB,iBAAiB,oRClC9B,4nEAmEA;2FDjCa,iBAAiB;kBAL7B,SAAS;mBAAC;oBACT,QAAQ,EAAE,aAAa;oBACvB,WAAW,EAAE,2BAA2B;oBACxC,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;+HAKU,KAAK;sBAAb,KAAK;gBAQF,QAAQ;sBADX,KAAK;uBAAC,OAAO;gBAWL,QAAQ;sBAAhB,KAAK;gBASG,SAAS;sBAAjB,KAAK;gBAKG,cAAc;sBAAtB,KAAK;gBAEG,aAAa;sBAArB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,QAAQ;sBAAhB,KAAK","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  Input,\n  isDevMode,\n  OnInit,\n  TemplateRef,\n} from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { ICON_TYPE } from '../../../cms-components/misc/icon/icon.model';\nimport { CarouselService } from './carousel.service';\n\n/**\n * Generic carousel component that can be used to render any carousel items,\n * such as products, images, banners, or any component. Carousel items are\n * rendered in so-called carousel slides, and the previous/next buttons as well as\n * the indicator-buttons can used to navigate the slides.\n *\n * The component uses an array of Observables (`items$`) as an input, to allow\n * for lazy loading of items.\n *\n * The number of items per slide is calculated with the `itemWidth`, which can given\n * in pixels or percentage.\n *\n * To allow for flexible rendering of items, the rendering is delegated to the\n * given `template`. This allows for maximum flexibility.\n */\n@Component({\n  selector: 'cx-carousel',\n  templateUrl: './carousel.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CarouselComponent implements OnInit {\n  /**\n   * The title is rendered as the carousel heading.\n   */\n  @Input() title: string;\n\n  /**\n   * The items$ represent the carousel items. The items$ are\n   * observables so that the items can be loaded on demand.\n   */\n  items: Observable<any>[];\n  @Input('items')\n  set setItems(inputItems: Observable<any>[]) {\n    this.items = inputItems;\n    //Reset slider when changing products\n    this.activeSlide = 0;\n  }\n\n  /**\n   * The template is rendered for each item, so that the actual\n   * view can be given by the compoent that uses the `CarouselComponent`.\n   */\n  @Input() template: TemplateRef<any>;\n\n  /**\n   * Specifies the minimum size of the carousel item, either in px or %.\n   * This value is used for the calculation of numbers per carousel, so that\n   * the number of carousel items is dynamic. The calculation uses the `itemWidth`\n   * and the host element `clientWidth`, so that the carousel is reusable in\n   * different layouts (for example in a 50% grid).\n   */\n  @Input() itemWidth = '300px';\n\n  /**\n   * Indicates whether the visual indicators are used.\n   */\n  @Input() hideIndicators = false;\n\n  @Input() indicatorIcon = ICON_TYPE.CIRCLE;\n  @Input() previousIcon = ICON_TYPE.CARET_LEFT;\n  @Input() nextIcon = ICON_TYPE.CARET_RIGHT;\n\n  activeSlide: number;\n  size$: Observable<number>;\n\n  constructor(protected el: ElementRef, protected service: CarouselService) {}\n\n  ngOnInit() {\n    if (!this.template && isDevMode()) {\n      console.error(\n        'No template reference provided to render the carousel items for the `cx-carousel`'\n      );\n      return;\n    }\n    this.size$ = this.service\n      .getItemsPerSlide(this.el.nativeElement, this.itemWidth)\n      .pipe(tap(() => (this.activeSlide = 0)));\n  }\n\n  getSlideNumber(size: number, currentIndex: number): number {\n    let normalizedCurrentIndex = currentIndex + 1;\n    return Math.ceil(normalizedCurrentIndex / size);\n  }\n}\n","<ng-container *ngIf=\"items?.length > 0 && (size$ | async) as size\">\n  <h2 *ngIf=\"title\">{{ title }}</h2>\n\n  <div class=\"carousel-panel\" [ngClass]=\"'size-' + size\">\n    <button\n      *ngIf=\"size < items.length\"\n      class=\"previous\"\n      (click)=\"activeSlide = activeSlide - size\"\n      [disabled]=\"activeSlide === 0\"\n      [attr.aria-label]=\"'carousel.previousSlide' | cxTranslate\"\n    >\n      <cx-icon [type]=\"previousIcon\"></cx-icon>\n    </button>\n\n    <div class=\"slides\">\n      <ng-container *ngFor=\"let _ of items; let i = index\">\n        <div\n          class=\"slide\"\n          *ngIf=\"i % size === 0\"\n          [class.active]=\"i === activeSlide\"\n        >\n          <ng-container\n            *ngFor=\"let item of items | slice: i:i + size; let j = index\"\n          >\n            <div\n              *ngIf=\"item | async as data\"\n              class=\"item\"\n              [class.active]=\"i === activeSlide\"\n            >\n              <ng-container\n                *ngTemplateOutlet=\"template; context: { item: data }\"\n              ></ng-container>\n            </div>\n          </ng-container>\n        </div>\n      </ng-container>\n    </div>\n\n    <button\n      *ngIf=\"size < items.length\"\n      class=\"next\"\n      (click)=\"activeSlide = activeSlide + size\"\n      tabindex=\"-1\"\n      [disabled]=\"activeSlide > items.length - size - 1\"\n      [attr.aria-label]=\"'carousel.nextSlide' | cxTranslate\"\n    >\n      <cx-icon [type]=\"nextIcon\"></cx-icon>\n    </button>\n  </div>\n\n  <div *ngIf=\"!hideIndicators && size < items.length\" class=\"indicators\">\n    <ng-container *ngFor=\"let _ of items; let i = index\">\n      <button\n        *ngIf=\"i % size === 0\"\n        (focus)=\"activeSlide = i\"\n        [disabled]=\"i === activeSlide\"\n        [attr.aria-label]=\"\n          'carousel.slideNumber'\n            | cxTranslate: { currentSlideNumber: getSlideNumber(size, i) }\n        \"\n        tabindex=\"-1\"\n      >\n        <cx-icon [type]=\"indicatorIcon\" aria-hidden=\"true\"></cx-icon>\n      </button>\n    </ng-container>\n  </div>\n</ng-container>\n"]}