UNPKG

@spartacus/storefront

Version:

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

153 lines 23.3 kB
import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Input, isDevMode, Output, } from '@angular/core'; import { TableLayout, } from './table.model'; import * as i0 from "@angular/core"; import * as i1 from "./table-renderer.service"; import * as i2 from "@angular/common"; import * as i3 from "../../../cms-structure/outlet/outlet.directive"; /** * The table component provides a generic table DOM structure, with 3 layout types: * horizontal, vertical and _stacked vertical_ layout. The layout is driven by the * table structure. * * The implementation is fairly "dumb" and only renders string based content for TH * and TD elements. The actual cell rendering is delegated to a (configurable) cell * component. Additionally, each cell is registered as an outlet, so that customizations * can be done by both outlet templates and components. * * The outlet references are concatenated from the table `type` and header `key`. The * following snippet shows an outlet generated for a table header, for the table type * "cost-center" with a header key "name": * * ``` * <th> * <template cxOutlet="table.cost-center.header.name"> * </template> * </th> * ``` * * Similarly, the data cells (`<td>`) are generated with the outlet template reference * `table.cost-center.data.name`. */ export class TableComponent { constructor(rendererService) { this.rendererService = rendererService; this.launch = new EventEmitter(); } set structure(structure) { this._structure = structure; this.init(); } get structure() { return this._structure; } init() { this.verticalLayout = !this.layout || this.layout === TableLayout.VERTICAL; this.verticalStackedLayout = this.layout === TableLayout.VERTICAL_STACKED; this.horizontalLayout = this.layout === TableLayout.HORIZONTAL; this.rendererService.add(this.structure); this.addTableDebugInfo(); } launchItem(item) { this.launch.emit(item); } /** * Indicates whether the given item is the current item. * * The current item is driven by the `currentItem`, that holds a * property and value to compare. */ isCurrentItem(item) { var _a, _b; if (!this.currentItem || !this.currentItem.value) { return false; } return ((_a = this.currentItem) === null || _a === void 0 ? void 0 : _a.value) === (item === null || item === void 0 ? void 0 : item[(_b = this.currentItem) === null || _b === void 0 ? void 0 : _b.property]); } /** * Returns the header (th) outlet reference for the given field. */ getHeaderOutletRef(field) { return this.rendererService.getHeaderOutletRef(this.type, field); } /** * Returns the header (th) outlet context for the given field. */ getHeaderOutletContext(field) { return this.rendererService.getHeaderOutletContext(this.type, this.options, this.i18nRoot, field); } /** * Returns the data (td) outlet reference for the given field. */ getDataOutletRef(field) { return this.rendererService.getDataOutletRef(this.type, field); } /** * Returns the data (td) outlet context for the given field. */ getDataOutletContext(field, data) { return this.rendererService.getDataOutletContext(this.type, this.options, this.i18nRoot, field, data); } trackData(_i, item) { return JSON.stringify(item); } /** * Generates the table type into the UI in devMode, so that developers * can easily get the notion of the table type. */ addTableDebugInfo() { if (isDevMode() && this.type) { this.tableType = this.type; } } /** * Helper method to return the deeply nested orientation configuration. */ get layout() { var _a, _b; return (_b = (_a = this.structure) === null || _a === void 0 ? void 0 : _a.options) === null || _b === void 0 ? void 0 : _b.layout; } /** * Helper method to return the deeply nested type. */ get type() { var _a; return (_a = this.structure) === null || _a === void 0 ? void 0 : _a.type; } get options() { var _a; return (_a = this.structure) === null || _a === void 0 ? void 0 : _a.options; } } TableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: TableComponent, deps: [{ token: i1.TableRendererService }], target: i0.ɵɵFactoryTarget.Component }); TableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: TableComponent, selector: "cx-table", inputs: { structure: "structure", data: "data", i18nRoot: "i18nRoot", currentItem: "currentItem" }, outputs: { launch: "launch" }, host: { properties: { "attr.__cx-table-type": "this.tableType", "class.horizontal": "this.horizontalLayout", "class.vertical": "this.verticalLayout", "class.vertical-stacked": "this.verticalStackedLayout" } }, ngImport: i0, template: "<table *ngIf=\"structure\">\n <ng-container *ngIf=\"verticalStackedLayout\">\n <tbody\n *ngFor=\"let item of data; trackBy: trackData\"\n (click)=\"launchItem(item)\"\n [class.is-current]=\"isCurrentItem(item)\"\n >\n <tr *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n <th>\n <ng-template\n [cxOutlet]=\"getHeaderOutletRef(cell)\"\n [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n >\n {{ cell }}\n </ng-template>\n </th>\n <td>\n <ng-template\n [cxOutlet]=\"getDataOutletRef(cell)\"\n [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n >\n {{ item[cell] }}\n </ng-template>\n </td>\n </tr>\n </tbody>\n </ng-container>\n\n <!-- vertical tables render the item -->\n <ng-container *ngIf=\"verticalLayout\">\n <thead>\n <tr>\n <th scope=\"col\" *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n <ng-template\n [cxOutlet]=\"getHeaderOutletRef(cell)\"\n [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n >\n {{ cell }}\n </ng-template>\n </th>\n </tr>\n </thead>\n\n <tr\n *ngFor=\"let item of data; trackBy: trackData\"\n [class.is-current]=\"isCurrentItem(item)\"\n (click)=\"launchItem(item)\"\n >\n <td *ngFor=\"let cell of structure.cells; let i = index\" [class]=\"cell\">\n <ng-template\n [cxOutlet]=\"getDataOutletRef(cell)\"\n [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n >\n {{ item[cell] }}\n </ng-template>\n </td>\n </tr>\n </ng-container>\n\n <ng-container *ngIf=\"horizontalLayout\">\n <tr *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n <th scope=\"col\">\n <ng-template\n [cxOutlet]=\"getHeaderOutletRef(cell)\"\n [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n >\n {{ cell }}\n </ng-template>\n </th>\n <td\n *ngFor=\"let item of data; trackBy: trackData\"\n [class.is-current]=\"isCurrentItem(item)\"\n (click)=\"launchItem(item)\"\n >\n <ng-template\n [cxOutlet]=\"getDataOutletRef(cell)\"\n [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n >\n {{ item[cell] }}\n </ng-template>\n </td>\n </tr>\n </ng-container>\n</table>\n", directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.OutletDirective, selector: "[cxOutlet]", inputs: ["cxOutlet", "cxOutletContext", "cxOutletDefer"], outputs: ["loaded"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: TableComponent, decorators: [{ type: Component, args: [{ selector: 'cx-table', templateUrl: './table.component.html', changeDetection: ChangeDetectionStrategy.OnPush, }] }], ctorParameters: function () { return [{ type: i1.TableRendererService }]; }, propDecorators: { tableType: [{ type: HostBinding, args: ['attr.__cx-table-type'] }], horizontalLayout: [{ type: HostBinding, args: ['class.horizontal'] }], verticalLayout: [{ type: HostBinding, args: ['class.vertical'] }], verticalStackedLayout: [{ type: HostBinding, args: ['class.vertical-stacked'] }], structure: [{ type: Input }], data: [{ type: Input }], i18nRoot: [{ type: Input }], currentItem: [{ type: Input }], launch: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/shared/components/table/table.component.ts","../../../../../../projects/storefrontlib/shared/components/table/table.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,YAAY,EACZ,WAAW,EACX,KAAK,EACL,SAAS,EACT,MAAM,GACP,MAAM,eAAe,CAAC;AAEvB,OAAO,EAGL,WAAW,GAEZ,MAAM,eAAe,CAAC;;;;;AAEvB;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAMH,MAAM,OAAO,cAAc;IAiCzB,YAAsB,eAAqC;QAArC,oBAAe,GAAf,eAAe,CAAsB;QAFjD,WAAM,GAAG,IAAI,YAAY,EAAE,CAAC;IAEwB,CAAC;IA1B/D,IAAa,SAAS,CAAC,SAAyB;QAC9C,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;QAC5B,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IACD,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAsBD,IAAI;QACF,IAAI,CAAC,cAAc,GAAG,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,KAAK,WAAW,CAAC,QAAQ,CAAC;QAC3E,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,MAAM,KAAK,WAAW,CAAC,gBAAgB,CAAC;QAC1E,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,MAAM,KAAK,WAAW,CAAC,UAAU,CAAC;QAE/D,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAEzC,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3B,CAAC;IAED,UAAU,CAAC,IAAS;QAClB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAED;;;;;OAKG;IACH,aAAa,CAAC,IAAS;;QACrB,IAAI,CAAC,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE;YAChD,OAAO,KAAK,CAAC;SACd;QACD,OAAO,CAAA,MAAA,IAAI,CAAC,WAAW,0CAAE,KAAK,OAAK,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAG,MAAA,IAAI,CAAC,WAAW,0CAAE,QAAQ,CAAC,CAAA,CAAC;IACxE,CAAC;IAED;;OAEG;IACH,kBAAkB,CAAC,KAAa;QAC9B,OAAO,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IACnE,CAAC;IAED;;OAEG;IACH,sBAAsB,CAAC,KAAa;QAClC,OAAO,IAAI,CAAC,eAAe,CAAC,sBAAsB,CAChD,IAAI,CAAC,IAAI,EACT,IAAI,CAAC,OAAO,EACZ,IAAI,CAAC,QAAQ,EACb,KAAK,CACN,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,gBAAgB,CAAC,KAAa;QAC5B,OAAO,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,oBAAoB,CAAC,KAAa,EAAE,IAAS;QAC3C,OAAO,IAAI,CAAC,eAAe,CAAC,oBAAoB,CAC9C,IAAI,CAAC,IAAI,EACT,IAAI,CAAC,OAAO,EACZ,IAAI,CAAC,QAAQ,EACb,KAAK,EACL,IAAI,CACL,CAAC;IACJ,CAAC;IAED,SAAS,CAAC,EAAU,EAAE,IAAS;QAC7B,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACO,iBAAiB;QACzB,IAAI,SAAS,EAAE,IAAI,IAAI,CAAC,IAAI,EAAE;YAC5B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC;SAC5B;IACH,CAAC;IAED;;OAEG;IACH,IAAY,MAAM;;QAChB,OAAO,MAAA,MAAA,IAAI,CAAC,SAAS,0CAAE,OAAO,0CAAE,MAAM,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,IAAY,IAAI;;QACd,OAAO,MAAA,IAAI,CAAC,SAAS,0CAAE,IAAI,CAAC;IAC9B,CAAC;IAED,IAAY,OAAO;;QACjB,OAAO,MAAA,IAAI,CAAC,SAAS,0CAAE,OAAO,CAAC;IACjC,CAAC;;2GAnIU,cAAc;+FAAd,cAAc,qYC9C3B,29EAoFA;2FDtCa,cAAc;kBAL1B,SAAS;mBAAC;oBACT,QAAQ,EAAE,UAAU;oBACpB,WAAW,EAAE,wBAAwB;oBACrC,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;2GAEsC,SAAS;sBAA7C,WAAW;uBAAC,sBAAsB;gBACF,gBAAgB;sBAAhD,WAAW;uBAAC,kBAAkB;gBACA,cAAc;sBAA5C,WAAW;uBAAC,gBAAgB;gBACU,qBAAqB;sBAA3D,WAAW;uBAAC,wBAAwB;gBAGxB,SAAS;sBAArB,KAAK;gBAQG,IAAI;sBAAZ,KAAK;gBAMG,QAAQ;sBAAhB,KAAK;gBAQG,WAAW;sBAAnB,KAAK;gBAEI,MAAM;sBAAf,MAAM","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  HostBinding,\n  Input,\n  isDevMode,\n  Output,\n} from '@angular/core';\nimport { TableRendererService } from './table-renderer.service';\nimport {\n  TableDataOutletContext,\n  TableHeaderOutletContext,\n  TableLayout,\n  TableStructure,\n} from './table.model';\n\n/**\n * The table component provides a generic table DOM structure, with 3 layout types:\n * horizontal, vertical and _stacked vertical_ layout. The layout is driven by the\n * table structure.\n *\n * The implementation is fairly \"dumb\" and only renders string based content for TH\n * and TD elements. The actual cell rendering is delegated to a (configurable) cell\n * component. Additionally, each cell is registered as an outlet, so that customizations\n * can be done by both outlet templates and components.\n *\n * The outlet references are concatenated from the table `type` and header `key`. The\n * following snippet shows an outlet generated for a table header, for the table type\n * \"cost-center\" with a header key \"name\":\n *\n * ```\n * <th>\n *   <template cxOutlet=\"table.cost-center.header.name\">\n *   </template>\n * </th>\n * ```\n *\n * Similarly, the data cells (`<td>`) are generated with the outlet template reference\n * `table.cost-center.data.name`.\n */\n@Component({\n  selector: 'cx-table',\n  templateUrl: './table.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TableComponent<T> {\n  @HostBinding('attr.__cx-table-type') tableType: string;\n  @HostBinding('class.horizontal') horizontalLayout: boolean;\n  @HostBinding('class.vertical') verticalLayout: boolean;\n  @HostBinding('class.vertical-stacked') verticalStackedLayout: boolean;\n\n  private _structure: TableStructure;\n  @Input() set structure(structure: TableStructure) {\n    this._structure = structure;\n    this.init();\n  }\n  get structure(): TableStructure {\n    return this._structure;\n  }\n\n  @Input() data: T[];\n\n  /**\n   * The i18nRoot is passed into the table cell context, so that\n   * cell components can concatenate the i18n root and label.\n   */\n  @Input() i18nRoot: string;\n\n  /**\n   * Provides a mechanism to compare a matching value for each item.\n   *\n   * The `property` refers to the dataset.value property, and the value tot the\n   * matching property value.\n   */\n  @Input() currentItem: { value: any; property: string };\n\n  @Output() launch = new EventEmitter();\n\n  constructor(protected rendererService: TableRendererService) {}\n\n  init() {\n    this.verticalLayout = !this.layout || this.layout === TableLayout.VERTICAL;\n    this.verticalStackedLayout = this.layout === TableLayout.VERTICAL_STACKED;\n    this.horizontalLayout = this.layout === TableLayout.HORIZONTAL;\n\n    this.rendererService.add(this.structure);\n\n    this.addTableDebugInfo();\n  }\n\n  launchItem(item: any): void {\n    this.launch.emit(item);\n  }\n\n  /**\n   * Indicates whether the given item is the current item.\n   *\n   * The current item is driven by the `currentItem`, that holds a\n   * property and value to compare.\n   */\n  isCurrentItem(item: any): boolean {\n    if (!this.currentItem || !this.currentItem.value) {\n      return false;\n    }\n    return this.currentItem?.value === item?.[this.currentItem?.property];\n  }\n\n  /**\n   * Returns the header (th) outlet reference for the given field.\n   */\n  getHeaderOutletRef(field: string): string {\n    return this.rendererService.getHeaderOutletRef(this.type, field);\n  }\n\n  /**\n   * Returns the header (th) outlet context for the given field.\n   */\n  getHeaderOutletContext(field: string): TableHeaderOutletContext {\n    return this.rendererService.getHeaderOutletContext(\n      this.type,\n      this.options,\n      this.i18nRoot,\n      field\n    );\n  }\n\n  /**\n   * Returns the data (td) outlet reference for the given field.\n   */\n  getDataOutletRef(field: string): string {\n    return this.rendererService.getDataOutletRef(this.type, field);\n  }\n\n  /**\n   * Returns the data (td) outlet context for the given field.\n   */\n  getDataOutletContext(field: string, data: any): TableDataOutletContext {\n    return this.rendererService.getDataOutletContext(\n      this.type,\n      this.options,\n      this.i18nRoot,\n      field,\n      data\n    );\n  }\n\n  trackData(_i: number, item: any): any {\n    return JSON.stringify(item);\n  }\n\n  /**\n   * Generates the table type into the UI in devMode, so that developers\n   * can easily get the notion of the table type.\n   */\n  protected addTableDebugInfo() {\n    if (isDevMode() && this.type) {\n      this.tableType = this.type;\n    }\n  }\n\n  /**\n   * Helper method to return the deeply nested orientation configuration.\n   */\n  private get layout() {\n    return this.structure?.options?.layout;\n  }\n\n  /**\n   * Helper method to return the deeply nested type.\n   */\n  private get type() {\n    return this.structure?.type;\n  }\n\n  private get options() {\n    return this.structure?.options;\n  }\n}\n","<table *ngIf=\"structure\">\n  <ng-container *ngIf=\"verticalStackedLayout\">\n    <tbody\n      *ngFor=\"let item of data; trackBy: trackData\"\n      (click)=\"launchItem(item)\"\n      [class.is-current]=\"isCurrentItem(item)\"\n    >\n      <tr *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n        <th>\n          <ng-template\n            [cxOutlet]=\"getHeaderOutletRef(cell)\"\n            [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n          >\n            {{ cell }}\n          </ng-template>\n        </th>\n        <td>\n          <ng-template\n            [cxOutlet]=\"getDataOutletRef(cell)\"\n            [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n          >\n            {{ item[cell] }}\n          </ng-template>\n        </td>\n      </tr>\n    </tbody>\n  </ng-container>\n\n  <!-- vertical tables render the item  -->\n  <ng-container *ngIf=\"verticalLayout\">\n    <thead>\n      <tr>\n        <th scope=\"col\" *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n          <ng-template\n            [cxOutlet]=\"getHeaderOutletRef(cell)\"\n            [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n          >\n            {{ cell }}\n          </ng-template>\n        </th>\n      </tr>\n    </thead>\n\n    <tr\n      *ngFor=\"let item of data; trackBy: trackData\"\n      [class.is-current]=\"isCurrentItem(item)\"\n      (click)=\"launchItem(item)\"\n    >\n      <td *ngFor=\"let cell of structure.cells; let i = index\" [class]=\"cell\">\n        <ng-template\n          [cxOutlet]=\"getDataOutletRef(cell)\"\n          [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n        >\n          {{ item[cell] }}\n        </ng-template>\n      </td>\n    </tr>\n  </ng-container>\n\n  <ng-container *ngIf=\"horizontalLayout\">\n    <tr *ngFor=\"let cell of structure.cells\" [class]=\"cell\">\n      <th scope=\"col\">\n        <ng-template\n          [cxOutlet]=\"getHeaderOutletRef(cell)\"\n          [cxOutletContext]=\"getHeaderOutletContext(cell)\"\n        >\n          {{ cell }}\n        </ng-template>\n      </th>\n      <td\n        *ngFor=\"let item of data; trackBy: trackData\"\n        [class.is-current]=\"isCurrentItem(item)\"\n        (click)=\"launchItem(item)\"\n      >\n        <ng-template\n          [cxOutlet]=\"getDataOutletRef(cell)\"\n          [cxOutletContext]=\"getDataOutletContext(cell, item)\"\n        >\n          {{ item[cell] }}\n        </ng-template>\n      </td>\n    </tr>\n  </ng-container>\n</table>\n"]}