UNPKG

@spartacus/storefront

Version:

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

112 lines 14.9 kB
import { Component, HostBinding, HostListener, Input, ViewChild, } from '@angular/core'; import { startWith } from 'rxjs/operators'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; import * as i2 from "@spartacus/core"; /** * Provides a UI to manage the count of the quantity, typically by using * increase and decrease functionality. The item counter expects an input `FormControl` * so that the state of the control can be managed outside of this component. */ export class ItemCounterComponent { constructor() { /** * This can be used in case an item has a minmum order quantity. * @default 1 */ this.min = 1; /** * The step is used to increment the count. It is supposed to be a * positive integer or float. * @default 1 */ this.step = 1; /** * Indicates that the input can be manually set to zero, * despite the fact that the input controls will be limited to * the minimum. The zero value can be used to remove an item. */ this.allowZero = false; /** * In readonly mode the item counter will only be shown as a label, * the form controls are not rendered. * Please not that readonly is different from the `disabled` form state. * @default false */ this.readonly = false; } handleClick() { this.input.nativeElement.focus(); } ngOnInit() { this.sub = this.control.valueChanges .pipe(startWith(this.control.value)) .subscribe((value) => this.control.setValue(this.getValidCount(value), { emitEvent: false })); } ngOnDestroy() { if (this.sub) { this.sub.unsubscribe(); } } increment() { // it's too early to use the `stepUp` and `stepDown` API... // let's wait for FF: https://caniuse.com/#search=stepUp this.control.setValue(this.control.value + this.step); this.control.markAsDirty(); } decrement() { this.control.setValue(this.control.value - this.step); this.control.markAsDirty(); } /** * Validate that the given value is in between * the `min` and `max` value. If the value is out * of the min/max range, it will be altered. * If `allowZero` is set to true, the 0 value is ignored. * */ getValidCount(value) { if (value < this.min && !(value === 0 && this.allowZero)) { value = this.min; } if (this.max && value > this.max) { value = this.max; } return value; } } ItemCounterComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ItemCounterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); ItemCounterComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: ItemCounterComponent, selector: "cx-item-counter", inputs: { control: "control", min: "min", max: "max", step: "step", allowZero: "allowZero", readonly: "readonly" }, host: { listeners: { "click": "handleClick()" }, properties: { "class.readonly": "this.readonly" } }, viewQueries: [{ propertyName: "input", first: true, predicate: ["qty"], descendants: true }], ngImport: i0, template: "<button\n type=\"button\"\n (click)=\"decrement()\"\n [disabled]=\"control.disabled || control.value <= min\"\n [tabindex]=\"control.disabled || control.value <= min ? -1 : 0\"\n attr.aria-label=\"{{ 'itemCounter.removeOne' | cxTranslate }}\"\n>\n -\n</button>\n<input\n #qty\n type=\"number\"\n [min]=\"min\"\n [max]=\"max\"\n [step]=\"step\"\n [readonly]=\"readonly\"\n [tabindex]=\"readonly ? -1 : 0\"\n [formControl]=\"control\"\n attr.aria-label=\"{{ 'itemCounter.quantity' | cxTranslate }}\"\n/>\n<button\n type=\"button\"\n (click)=\"increment()\"\n [disabled]=\"control.disabled || control.value >= max\"\n tabindex=\"0\"\n attr.aria-label=\"{{ 'itemCounter.addOneMore' | cxTranslate }}\"\n>\n +\n</button>\n", directives: [{ type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { type: i1.MaxValidator, selector: "input[type=number][max][formControlName],input[type=number][max][formControl],input[type=number][max][ngModel]", inputs: ["max"] }, { type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }], pipes: { "cxTranslate": i2.TranslatePipe } }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: ItemCounterComponent, decorators: [{ type: Component, args: [{ selector: 'cx-item-counter', templateUrl: './item-counter.component.html', // do not use OnPush change detection strategy as we would not // get updates of other form control state (disabled). We want to have a // disabled state in order to ensure that the control cannot be used while // the cart is updated. }] }], propDecorators: { control: [{ type: Input }], min: [{ type: Input }], max: [{ type: Input }], step: [{ type: Input }], allowZero: [{ type: Input }], readonly: [{ type: HostBinding, args: ['class.readonly'] }, { type: Input }], input: [{ type: ViewChild, args: ['qty'] }], handleClick: [{ type: HostListener, args: ['click'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"item-counter.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/shared/components/item-counter/item-counter.component.ts","../../../../../../projects/storefrontlib/shared/components/item-counter/item-counter.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EAET,WAAW,EACX,YAAY,EACZ,KAAK,EAGL,SAAS,GACV,MAAM,eAAe,CAAC;AAGvB,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;;;;AAE3C;;;;GAIG;AASH,MAAM,OAAO,oBAAoB;IARjC;QAeE;;;WAGG;QACM,QAAG,GAAG,CAAC,CAAC;QAOjB;;;;WAIG;QACM,SAAI,GAAG,CAAC,CAAC;QAElB;;;;WAIG;QACM,cAAS,GAAG,KAAK,CAAC;QAE3B;;;;;WAKG;QACqC,aAAQ,GAAG,KAAK,CAAC;KAuD1D;IA9CwB,WAAW;QAChC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;IACnC,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY;aACjC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;aACnC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CACnB,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CACvE,CAAC;IACN,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;SACxB;IACH,CAAC;IAED,SAAS;QACP,2DAA2D;QAC3D,wDAAwD;QACxD,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC;QACtD,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;IAC7B,CAAC;IAED,SAAS;QACP,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC;QACtD,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;IAC7B,CAAC;IAED;;;;;;OAMG;IACK,aAAa,CAAC,KAAa;QACjC,IAAI,KAAK,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC,CAAC,KAAK,KAAK,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,EAAE;YACxD,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC;SAClB;QACD,IAAI,IAAI,CAAC,GAAG,IAAI,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE;YAChC,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC;SAClB;QACD,OAAO,KAAK,CAAC;IACf,CAAC;;iHA5FU,oBAAoB;qGAApB,oBAAoB,+WC3BjC,muBA6BA;2FDFa,oBAAoB;kBARhC,SAAS;mBAAC;oBACT,QAAQ,EAAE,iBAAiB;oBAC3B,WAAW,EAAE,+BAA+B;oBAC5C,8DAA8D;oBAC9D,wEAAwE;oBACxE,0EAA0E;oBAC1E,uBAAuB;iBACxB;8BAMU,OAAO;sBAAf,KAAK;gBAMG,GAAG;sBAAX,KAAK;gBAKG,GAAG;sBAAX,KAAK;gBAOG,IAAI;sBAAZ,KAAK;gBAOG,SAAS;sBAAjB,KAAK;gBAQkC,QAAQ;sBAA/C,WAAW;uBAAC,gBAAgB;;sBAAG,KAAK;gBAEX,KAAK;sBAA9B,SAAS;uBAAC,KAAK;gBAOO,WAAW;sBAAjC,YAAY;uBAAC,OAAO","sourcesContent":["import {\n  Component,\n  ElementRef,\n  HostBinding,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { Subscription } from 'rxjs';\nimport { startWith } from 'rxjs/operators';\n\n/**\n * Provides a UI to manage the count of the quantity, typically by using\n * increase and decrease functionality. The item counter expects an input `FormControl`\n * so that the state of the control can be managed outside of this component.\n */\n@Component({\n  selector: 'cx-item-counter',\n  templateUrl: './item-counter.component.html',\n  // do not use OnPush change detection strategy as we would not\n  // get updates of other form control state (disabled). We want to have a\n  // disabled state in order to ensure that the control cannot be used while\n  // the cart is updated.\n})\nexport class ItemCounterComponent implements OnInit, OnDestroy {\n  /**\n   * Holds the value of the counter, the state of the `FormControl`\n   * can be managed outside of the item counter.\n   */\n  @Input() control: FormControl;\n\n  /**\n   * This can be used in case an item has a minmum order quantity.\n   * @default 1\n   */\n  @Input() min = 1;\n\n  /**\n   * This can be used in case an item has a maximum order quantity.\n   */\n  @Input() max: number;\n\n  /**\n   * The step is used to increment the count. It is supposed to be a\n   * positive integer or float.\n   * @default 1\n   */\n  @Input() step = 1;\n\n  /**\n   * Indicates that the input can be manually set to zero,\n   * despite the fact that the input controls will be limited to\n   * the minimum. The zero value can be used to remove an item.\n   */\n  @Input() allowZero = false;\n\n  /**\n   * In readonly mode the item counter will only be shown as a label,\n   * the form controls are not rendered.\n   * Please not that readonly is different from the `disabled` form state.\n   * @default false\n   */\n  @HostBinding('class.readonly') @Input() readonly = false;\n\n  @ViewChild('qty') private input: ElementRef<HTMLInputElement>;\n\n  /**\n   * Subscription responsible for auto-correcting control's value when it's invalid.\n   */\n  private sub: Subscription;\n\n  @HostListener('click') handleClick() {\n    this.input.nativeElement.focus();\n  }\n\n  ngOnInit() {\n    this.sub = this.control.valueChanges\n      .pipe(startWith(this.control.value))\n      .subscribe((value) =>\n        this.control.setValue(this.getValidCount(value), { emitEvent: false })\n      );\n  }\n\n  ngOnDestroy() {\n    if (this.sub) {\n      this.sub.unsubscribe();\n    }\n  }\n\n  increment() {\n    // it's too early to use the `stepUp` and `stepDown` API...\n    // let's wait for FF: https://caniuse.com/#search=stepUp\n    this.control.setValue(this.control.value + this.step);\n    this.control.markAsDirty();\n  }\n\n  decrement() {\n    this.control.setValue(this.control.value - this.step);\n    this.control.markAsDirty();\n  }\n\n  /**\n   * Validate that the given value is in between\n   * the `min` and `max` value. If the value is out\n   * of  the min/max range, it will be altered.\n   * If `allowZero` is set to true, the 0 value is ignored.\n   *\n   */\n  private getValidCount(value: number) {\n    if (value < this.min && !(value === 0 && this.allowZero)) {\n      value = this.min;\n    }\n    if (this.max && value > this.max) {\n      value = this.max;\n    }\n    return value;\n  }\n}\n","<button\n  type=\"button\"\n  (click)=\"decrement()\"\n  [disabled]=\"control.disabled || control.value <= min\"\n  [tabindex]=\"control.disabled || control.value <= min ? -1 : 0\"\n  attr.aria-label=\"{{ 'itemCounter.removeOne' | cxTranslate }}\"\n>\n  -\n</button>\n<input\n  #qty\n  type=\"number\"\n  [min]=\"min\"\n  [max]=\"max\"\n  [step]=\"step\"\n  [readonly]=\"readonly\"\n  [tabindex]=\"readonly ? -1 : 0\"\n  [formControl]=\"control\"\n  attr.aria-label=\"{{ 'itemCounter.quantity' | cxTranslate }}\"\n/>\n<button\n  type=\"button\"\n  (click)=\"increment()\"\n  [disabled]=\"control.disabled || control.value >= max\"\n  tabindex=\"0\"\n  attr.aria-label=\"{{ 'itemCounter.addOneMore' | cxTranslate }}\"\n>\n  +\n</button>\n"]}