@energycap/components 0.39.31 → 0.39.32-ECAP-26866-table-multi-sort-improvements.20241022-1702

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,5 @@
1
1
  import { Component, ContentChildren, EventEmitter, HostBinding, Input, Output, ViewEncapsulation } from '@angular/core';
2
- import { isEqual } from 'lodash';
2
+ import { cloneDeep, isEqual } from 'lodash';
3
3
  import { Subject } from 'rxjs';
4
4
  import { takeUntil } from 'rxjs/operators';
5
5
  import { ResizableColumnComponent } from './resizable-column.component';
@@ -77,7 +77,7 @@ export class TableComponent {
77
77
  if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {
78
78
  const newValue = changes.sort.currentValue;
79
79
  if (!this.currentSort || !isEqual(newValue, this.currentSort)) {
80
- this.onSort(newValue.field, newValue.direction);
80
+ this.onSort(newValue);
81
81
  }
82
82
  }
83
83
  }
@@ -126,7 +126,14 @@ export class TableComponent {
126
126
  if (th.classList.contains('is-resizable') && event.target && event.target.classList.contains('handle')) {
127
127
  return;
128
128
  }
129
- this.onSort(field);
129
+ let direction;
130
+ if (th.classList.contains('is-sorted-desc')) {
131
+ direction = 'asc';
132
+ }
133
+ else {
134
+ direction = 'desc';
135
+ }
136
+ this.onSort({ field, direction });
130
137
  });
131
138
  // Store unlisten function to remove listener when component is destroyed
132
139
  this.unlisteners.push(unlisten);
@@ -138,7 +145,7 @@ export class TableComponent {
138
145
  });
139
146
  // Set the initial sort
140
147
  if (this.sort) {
141
- this.onSort(this.sort.field, this.sort.direction);
148
+ this.onSort(this.sort);
142
149
  }
143
150
  }
144
151
  /**
@@ -147,32 +154,29 @@ export class TableComponent {
147
154
  * Emit the current field and direction of the sorted column
148
155
  * @param field
149
156
  */
150
- onSort(field, direction) {
151
- const th = this.sortableCols[field];
152
- if (!th) {
153
- return;
154
- }
155
- if (this.currentSort && this.currentSort.field !== field) {
156
- this.renderer.removeClass(this.sortableCols[this.currentSort.field], `is-sorted-${this.currentSort.direction}`);
157
- }
158
- if (th.classList.contains('is-sorted-desc')) {
159
- this.renderer.removeClass(th, 'is-sorted-desc');
160
- if (!direction) {
161
- direction = 'asc';
162
- }
157
+ onSort(sort) {
158
+ this.clearSort();
159
+ if (Array.isArray(sort)) {
160
+ sort.forEach(s => this.applySort(s));
163
161
  }
164
162
  else {
165
- if (th.classList.contains('is-sorted-asc')) {
166
- this.renderer.removeClass(th, 'is-sorted-asc');
167
- }
168
- if (!direction) {
169
- direction = 'desc';
170
- }
163
+ this.applySort(sort);
164
+ }
165
+ this.currentSort = cloneDeep(sort);
166
+ this.sortChange.emit(this.currentSort);
167
+ }
168
+ clearSort() {
169
+ Object.values(this.sortableCols).forEach(th => {
170
+ this.renderer.removeClass(th, `is-sorted-asc`);
171
+ this.renderer.removeClass(th, `is-sorted-desc`);
172
+ });
173
+ }
174
+ applySort(sort) {
175
+ const th = this.sortableCols[sort.field];
176
+ if (!th) {
177
+ return;
171
178
  }
172
- this.renderer.addClass(th, `is-sorted-${direction}`);
173
- const sortEvent = { field: field, direction: direction };
174
- this.currentSort = sortEvent;
175
- this.sortChange.emit(sortEvent);
179
+ this.renderer.addClass(th, `is-sorted-${sort.direction}`);
176
180
  }
177
181
  initSelectable() {
178
182
  this.selectionContext.rowCheckboxes.valueChanges.pipe(takeUntil(this.destroyed)).subscribe((value) => {
@@ -241,4 +245,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
241
245
  type: Input,
242
246
  args: ['resizableColumns']
243
247
  }] } });
244
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/table/table.component.ts","../../../../../../projects/components/src/lib/display/table/table.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAmC,SAAS,EAAE,eAAe,EAAc,YAAY,EAAE,WAAW,EAAE,KAAK,EAAwB,MAAM,EAAoD,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAC7O,OAAO,EAAE,OAAO,EAAE,MAAM,QAAQ,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAC3C,OAAO,EAAE,wBAAwB,EAAE,MAAM,8BAA8B,CAAC;AACxE,OAAO,EAAE,uBAAuB,EAAE,MAAM,8BAA8B,CAAC;;;;AAevE,MAAM,OAAO,cAAc;IA4GzB,YAAoB,QAAmB,EAAS,EAAc;QAA1C,aAAQ,GAAR,QAAQ,CAAW;QAAS,OAAE,GAAF,EAAE,CAAY;QA1GrD,OAAE,GAAG,EAAE,CAAC;QAEjB;;;WAGG;QAEM,eAAU,GAAG,KAAK,CAAC;QAE5B;;;WAGG;QAEM,cAAS,GAAG,KAAK,CAAC;QAE3B;;;;WAIG;QAEM,cAAS,GAAG,KAAK,CAAC;QAE3B;;;;;;;;;;;;;;;;;;WAkBG;QAEM,aAAQ,GAAG,KAAK,CAAC;QAa1B,iDAAiD;QAExC,eAAU,GAAG,KAAK,CAAC;QAE5B,iFAAiF;QACjF,iBAAY,GAAG,KAAK,CAAA;QAEpB;;WAEG;QAEM,WAAM,GAAG,KAAK,CAAC;QAOxB;;WAEG;QACO,eAAU,GAA4B,IAAI,YAAY,EAAE,CAAC;QAEnE;;WAEG;QACoC,wBAAmB,GAAG,KAAK,CAAC;QASnE;;;WAGG;QACK,gBAAW,GAAmB,EAAE,CAAC;QAKzC,wCAAwC;QAChC,iBAAY,GAAkD,EAAE,CAAC;QAEjE,cAAS,GAAkB,IAAI,OAAO,EAAE,CAAC;IAEgB,CAAC;IAElE,WAAW,CAAC,OAAsB;QAChC,IAAI,IAAI,CAAC,QAAQ,IAAI,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE;YAC3F,MAAM,QAAQ,GAAc,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC;YACtD,IAAI,CAAC,IAAI,CAAC,WAAW,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,EAAC;gBAC5D,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,EAAE,QAAQ,CAAC,SAAS,CAAC,CAAC;aACjD;SACF;IACH,CAAC;IAED,kBAAkB;QAChB,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;QAEpD,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAC1B,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAC1B,CAAC,SAAS,CAAC,GAAG,EAAE;YACf,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,+BAA+B,IAAI,IAAI,CAAC,iBAAiB,CAAC;IACzF,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,YAAY,EAAE,CAAC;SACrB;QAED,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,IAAI,CAAC,cAAc,EAAE,CAAC;SACvB;IACH,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE;YAC3B,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;SAClD;QAED,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;QACtB,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACK,aAAa,CAAI,QAAgB;QACvC,MAAM,KAAK,GAAG,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC;QACpC,OAAO,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC,CAAC;IACtD,CAAC;IAED;;;OAGG;IACK,YAAY;QAClB,IAAI,CAAC,aAAa,CAA6B,IAAI,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE;YAChE,IAAI,EAAE,CAAC,YAAY,CAAC,gBAAgB,CAAC,EAAE;gBAErC,MAAM,KAAK,GAAG,EAAE,CAAC,YAAY,CAAC,gBAAgB,CAAC,CAAC;gBAChD,IAAI,KAAK,EAAE;oBACT,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;oBAC9B,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,EAAE,OAAO,EAAE,CAAC,KAAiB,EAAE,EAAE;wBAEvE,2DAA2D;wBAC3D,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,KAAK,CAAC,MAAM,IAAkB,KAAK,CAAC,MAAO,CAAC,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE;4BACrH,OAAO;yBACR;wBAED,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;oBACrB,CAAC,CAAC,CAAC;oBAEH,yEAAyE;oBACzE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBACjC;qBAAM;oBACL,OAAO,CAAC,KAAK,CAAC,oDAAoD,EAAE,EAAE,CAAC,CAAC;iBACzE;aACF;QACH,CAAC,CAAC,CAAC;QAEH,uBAAuB;QACvB,IAAI,IAAI,CAAC,IAAI,EAAE;YACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SACnD;IACH,CAAC;IAED;;;;;OAKG;IACK,MAAM,CAAC,KAAa,EAAE,SAA8B;QAC1D,MAAM,EAAE,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QACpC,IAAI,CAAC,EAAE,EAAE;YAAE,OAAO;SAAE;QAEpB,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,KAAK,KAAK,KAAK,EAAE;YACxD,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,aAAa,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,CAAC;SACjH;QAED,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,gBAAgB,CAAC,EAAE;YAC3C,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,EAAE,gBAAgB,CAAC,CAAC;YAEhD,IAAI,CAAC,SAAS,EAAE;gBACd,SAAS,GAAG,KAAK,CAAC;aACnB;SAEF;aAAM;YACL,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE;gBAC1C,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,EAAE,eAAe,CAAC,CAAC;aAChD;YAED,IAAI,CAAC,SAAS,EAAE;gBACd,SAAS,GAAG,MAAM,CAAC;aACpB;SACF;QAED,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,EAAE,aAAa,SAAS,EAAE,CAAC,CAAC;QAErD,MAAM,SAAS,GAAG,EAAC,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAC,CAAC;QACvD,IAAI,CAAC,WAAW,GAAG,SAAS,CAAC;QAC7B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAEO,cAAc;QACpB,IAAI,CAAC,gBAAiB,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CACpD,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAC1B,CAAC,SAAS,CAAC,CAAC,KAAgB,EAAE,EAAE;YAC/B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;IACL,CAAC;;2GA9OU,cAAc;+FAAd,cAAc,8yBAuFR,uBAAuB,uEAGvB,wBAAwB,qEC9G3C,mjBAcM;2FDMO,cAAc;kBAN1B,SAAS;+BACE,UAAU,iBAGL,iBAAiB,CAAC,IAAI;yHAI5B,EAAE;sBADV,WAAW;uBAAC,SAAS;;sBACrB,KAAK;gBAOG,UAAU;sBADlB,WAAW;uBAAC,qBAAqB;;sBACjC,KAAK;gBAOG,SAAS;sBADjB,WAAW;uBAAC,oBAAoB;;sBAChC,KAAK;gBAQG,SAAS;sBADjB,WAAW;uBAAC,oBAAoB;;sBAChC,KAAK;gBAsBG,QAAQ;sBADhB,WAAW;uBAAC,mBAAmB;;sBAC/B,KAAK;gBAQG,gBAAgB;sBAAxB,KAAK;gBAGG,wBAAwB;sBAAhC,KAAK;gBAIG,UAAU;sBADlB,WAAW;uBAAC,qBAAqB;;sBACjC,KAAK;gBASG,MAAM;sBADd,WAAW;uBAAC,qBAAqB;;sBACjC,KAAK;gBAKG,IAAI;sBAAZ,KAAK;gBAKI,UAAU;sBAAnB,MAAM;gBAKgC,mBAAmB;sBAAzD,WAAW;uBAAC,wBAAwB;gBAE0B,UAAU;sBAAxE,eAAe;uBAAC,uBAAuB,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC;gBAGG,iBAAiB;sBAAhF,eAAe;uBAAC,wBAAwB,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC;gBAEnC,+BAA+B;sBAAzD,KAAK;uBAAC,kBAAkB","sourcesContent":["import { AfterContentInit, AfterViewInit, Component, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, OnChanges, OnDestroy, Output, QueryList, Renderer2, SimpleChanges, TemplateRef, ViewEncapsulation } from '@angular/core';\r\nimport { isEqual } from 'lodash';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ResizableColumnComponent } from './resizable-column.component';\r\nimport { TableMasterRowComponent } from './table-master-row.component';\r\nimport { TableSelectableRowContext } from './table-selectable-row.component';\r\n\r\nexport type TableSortDirection = 'desc' | 'asc';\r\nexport interface TableSort {\r\n  field: string;\r\n  direction: TableSortDirection;\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-table',\r\n  templateUrl: './table.component.html',\r\n  styleUrls: ['./table.component.scss'],\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class TableComponent implements AfterViewInit, OnDestroy, OnChanges, AfterContentInit{\r\n  @HostBinding('attr.id')\r\n  @Input() id = '';\r\n\r\n  /**\r\n   * Enables table scrolling with a sticky header and footer. Use\r\n   * <thead> and <tfoot> elements for styles to be applied correctly. \r\n   */\r\n  @HostBinding('class.is-scrollable')\r\n  @Input() scrollable = false;\r\n  \r\n  /**\r\n   * Enables resizable table columns. Apply the 'is-resizable' class\r\n   * to the <th> of a table column than can be resized.\r\n   */\r\n  @HostBinding('class.is-resizable')\r\n  @Input() resizable = false;\r\n  \r\n  /** \r\n   * Turns on condensed table styling. By default a condensed table\r\n   * has no row borders. To show borders, apply the 'has-borders' class\r\n   * to the <ec-table> element\r\n   */\r\n  @HostBinding('class.is-condensed')\r\n  @Input() condensed = false;\r\n  \r\n  /** \r\n   * Enables sorting by clicking on table columns. To make a column sortable\r\n   * apply the `data-sortable` attribute to the columns `<th>` element with \r\n   * the value of the field used to sort the column. When a column is clicked\r\n   * the fields and current sort direction is emitted from the `sortChange`\r\n   * output\r\n   * \r\n   * @example\r\n   * ```html\r\n   * <ec-table [sortable]=\"true\" [sort]=\"currentSort\" (sortChange)=\"onSort($event)\">\r\n   *  <thead>\r\n   *    <th data-sortable=\"name\">Name</th>\r\n   *    <th data-sortable=\"qty\">Qty.</th>\r\n   *    <th data-sortable=\"price\">Price</th>\r\n   *  </thead>\r\n   *  ...\r\n   * </ec-table>\r\n   * ```\r\n   */\r\n  @HostBinding('class.is-sortable')\r\n  @Input() sortable = false;\r\n  \r\n  /**\r\n   * Shared context of TableSelectableRowComponents in the table. Will turn on highlighting on row hover.\r\n   * Add the `TableSelectableRowComponent` to the table content to add checkboxes to each row and assign a \r\n   * TemplateRef to the `[selectionToolbarTemplate]` input that includes actions to take on selected items \r\n   * that will appear over the table header when 1 or more rows are selected.\r\n   */\r\n  @Input() selectionContext?: TableSelectableRowContext;\r\n\r\n  /** Include actions to display in a bar over the table header when 1 or more rows are selected. Requires `[selectionContext]` to be defined */\r\n  @Input() selectionToolbarTemplate?: TemplateRef<any>;\r\n  \r\n  /** Set to true if selectionContext is defined */\r\n  @HostBinding('class.is-selectable')\r\n  @Input() selectable = false;\r\n\r\n  /** Hides/shows the selection toolbar template when 1 or more rows is selected */\r\n  hasSelection = false\r\n\r\n  /**\r\n   * Enables form table styling to allow form controls inside table cells.\r\n   */\r\n  @HostBinding('class.is-form-table')\r\n  @Input() isForm = false;\r\n  \r\n  /**\r\n   * The current sort field and direction ('desc' or 'asc')\r\n   */\r\n  @Input() sort?: TableSort;\r\n\r\n  /**\r\n   * Emits the new sort value whenever a sortable column is clicked\r\n   */\r\n  @Output() sortChange: EventEmitter<TableSort> = new EventEmitter();\r\n\r\n  /**\r\n   * True if a TableMasterRowComponent exists in the table content\r\n   */\r\n  @HostBinding('class.is-master-detail') hasMasterDetailRows = false;\r\n\r\n  @ContentChildren(TableMasterRowComponent, {descendants: true}) masterRows!: QueryList<TableMasterRowComponent>;\r\n\r\n  public resizableColumns!: QueryList<ResizableColumnComponent>;\r\n  @ContentChildren(ResizableColumnComponent, {descendants: true}) _resizableColumns!: QueryList<ResizableColumnComponent>;\r\n\r\n  @Input('resizableColumns') searchableTableResizableColumns?: QueryList<ResizableColumnComponent>;\r\n\r\n  /**\r\n   * Unlisten function returned by `renderer.listen()` that are called\r\n   * OnDestroy to clean up all listeners\r\n   */\r\n  private unlisteners: (() => void)[] = [];\r\n  \r\n  /** The current sort value */\r\n  private currentSort?: TableSort;\r\n  \r\n  /** A map of sortable `<th>` elements */\r\n  private sortableCols: {[field: string]: HTMLTableHeaderCellElement} = {};\r\n\r\n  private destroyed: Subject<void> = new Subject();\r\n  \r\n  constructor(private renderer: Renderer2, public el: ElementRef) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {\r\n      const newValue: TableSort = changes.sort.currentValue;\r\n      if (!this.currentSort || !isEqual(newValue, this.currentSort)){\r\n        this.onSort(newValue.field, newValue.direction);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngAfterContentInit(): void {\r\n    this.hasMasterDetailRows = !!this.masterRows.length;\r\n    \r\n    this.masterRows.changes.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(() => {\r\n      this.hasMasterDetailRows = !!this.masterRows.length;\r\n    });\r\n\r\n    this.resizableColumns = this.searchableTableResizableColumns || this._resizableColumns;\r\n  }\r\n  \r\n  ngAfterViewInit(): void {\r\n    if (this.sortable) {\r\n      this.initSortable();\r\n    }\r\n\r\n    if (this.selectable) {\r\n      this.initSelectable();\r\n    }\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    if (this.unlisteners.length) {\r\n      this.unlisteners.forEach(unlisten => unlisten());\r\n    }\r\n    \r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Get all elements inside the host element that match a selector \r\n   * @param selector - Any valid selector for [querySelectorAll](https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelectorAll)\r\n   */\r\n  private queryElements<T>(selector: string): T[] {\r\n    const table = this.el.nativeElement;\r\n    return Array.from(table.querySelectorAll(selector));\r\n  }\r\n\r\n  /**\r\n   * Store a `<th>` for each sortable column and attach a click\r\n   * event listener. \r\n   */\r\n  private initSortable(): void {\r\n    this.queryElements<HTMLTableHeaderCellElement>('th').forEach(th => {\r\n      if (th.hasAttribute('data-sortfield')) {\r\n        \r\n        const field = th.getAttribute('data-sortfield');\r\n        if (field) {\r\n          this.sortableCols[field] = th;\r\n          const unlisten = this.renderer.listen(th, 'click', (event: MouseEvent) => {\r\n\r\n            // Do not sort if the event target was the resizable handle\r\n            if (th.classList.contains('is-resizable') && event.target && (<HTMLElement>event.target).classList.contains('handle')) {\r\n              return;\r\n            }\r\n\r\n            this.onSort(field);\r\n          });\r\n  \r\n          // Store unlisten function to remove listener when component is destroyed\r\n          this.unlisteners.push(unlisten);\r\n        } else {\r\n          console.error(`data-sortfield attribute was not assigned a value.`, th);\r\n        }\r\n      }\r\n    });\r\n\r\n    // Set the initial sort\r\n    if (this.sort) {\r\n      this.onSort(this.sort.field, this.sort.direction);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Find the sorted column and apply the desc or asc class in order\r\n   * to show the correct sort icon. If another field was sorted, reset it.\r\n   * Emit the current field and direction of the sorted column\r\n   * @param field \r\n   */\r\n  private onSort(field: string, direction?: TableSortDirection): void {\r\n    const th = this.sortableCols[field];\r\n    if (!th) { return; }\r\n    \r\n    if (this.currentSort && this.currentSort.field !== field) {\r\n      this.renderer.removeClass(this.sortableCols[this.currentSort.field], `is-sorted-${this.currentSort.direction}`);\r\n    } \r\n    \r\n    if (th.classList.contains('is-sorted-desc')) {\r\n      this.renderer.removeClass(th, 'is-sorted-desc');\r\n\r\n      if (!direction) {\r\n        direction = 'asc';\r\n      }\r\n\r\n    } else {\r\n      if (th.classList.contains('is-sorted-asc')) {\r\n        this.renderer.removeClass(th, 'is-sorted-asc');\r\n      }\r\n\r\n      if (!direction) {\r\n        direction = 'desc';\r\n      }\r\n    }\r\n\r\n    this.renderer.addClass(th, `is-sorted-${direction}`);\r\n\r\n    const sortEvent = {field: field, direction: direction};\r\n    this.currentSort = sortEvent;\r\n    this.sortChange.emit(sortEvent);\r\n  }\r\n\r\n  private initSelectable(): void {\r\n    this.selectionContext!.rowCheckboxes.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe((value: boolean[]) => {\r\n      this.hasSelection = value.some(v => v);\r\n    });\r\n  }\r\n}","<div class=\"selectable-table-toolbar\"\r\n     *ngIf=\"hasSelection && selectionToolbarTemplate\">\r\n  <ng-container *ngTemplateOutlet=\"selectionToolbarTemplate\"></ng-container>\r\n</div>\r\n<div id=\"{{id}}-scroll-container\" \r\n     class=\"table-scroll-container\">\r\n  <table class=\"main-table\"\r\n         id=\"{{id}}\"\r\n         [ecResizableTable]=\"resizable\"\r\n         [containerEl]=\"el\"\r\n         [sortableTable]=\"sortable\"\r\n         [resizableColumns]=\"resizableColumns\">\r\n    <ng-content></ng-content>\r\n  </table>\r\n</div>"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/table/table.component.ts","../../../../../../projects/components/src/lib/display/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, AfterViewInit, Component, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, OnChanges, OnDestroy, Output, QueryList, Renderer2, SimpleChanges, TemplateRef, ViewEncapsulation } from '@angular/core';\r\nimport { cloneDeep, isEqual } from 'lodash';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ResizableColumnComponent } from './resizable-column.component';\r\nimport { TableMasterRowComponent } from './table-master-row.component';\r\nimport { TableSelectableRowContext } from './table-selectable-row.component';\r\n\r\nexport type TableSortDirection = 'desc' | 'asc';\r\nexport interface TableSort {\r\n  field: string;\r\n  direction: TableSortDirection;\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-table',\r\n  templateUrl: './table.component.html',\r\n  styleUrls: ['./table.component.scss'],\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class TableComponent implements AfterViewInit, OnDestroy, OnChanges, AfterContentInit{\r\n  @HostBinding('attr.id')\r\n  @Input() id = '';\r\n\r\n  /**\r\n   * Enables table scrolling with a sticky header and footer. Use\r\n   * <thead> and <tfoot> elements for styles to be applied correctly. \r\n   */\r\n  @HostBinding('class.is-scrollable')\r\n  @Input() scrollable = false;\r\n  \r\n  /**\r\n   * Enables resizable table columns. Apply the 'is-resizable' class\r\n   * to the <th> of a table column than can be resized.\r\n   */\r\n  @HostBinding('class.is-resizable')\r\n  @Input() resizable = false;\r\n  \r\n  /** \r\n   * Turns on condensed table styling. By default a condensed table\r\n   * has no row borders. To show borders, apply the 'has-borders' class\r\n   * to the <ec-table> element\r\n   */\r\n  @HostBinding('class.is-condensed')\r\n  @Input() condensed = false;\r\n  \r\n  /** \r\n   * Enables sorting by clicking on table columns. To make a column sortable\r\n   * apply the `data-sortable` attribute to the columns `<th>` element with \r\n   * the value of the field used to sort the column. When a column is clicked\r\n   * the fields and current sort direction is emitted from the `sortChange`\r\n   * output\r\n   * \r\n   * @example\r\n   * ```html\r\n   * <ec-table [sortable]=\"true\" [sort]=\"currentSort\" (sortChange)=\"onSort($event)\">\r\n   *  <thead>\r\n   *    <th data-sortable=\"name\">Name</th>\r\n   *    <th data-sortable=\"qty\">Qty.</th>\r\n   *    <th data-sortable=\"price\">Price</th>\r\n   *  </thead>\r\n   *  ...\r\n   * </ec-table>\r\n   * ```\r\n   */\r\n  @HostBinding('class.is-sortable')\r\n  @Input() sortable = false;\r\n  \r\n  /**\r\n   * Shared context of TableSelectableRowComponents in the table. Will turn on highlighting on row hover.\r\n   * Add the `TableSelectableRowComponent` to the table content to add checkboxes to each row and assign a \r\n   * TemplateRef to the `[selectionToolbarTemplate]` input that includes actions to take on selected items \r\n   * that will appear over the table header when 1 or more rows are selected.\r\n   */\r\n  @Input() selectionContext?: TableSelectableRowContext;\r\n\r\n  /** Include actions to display in a bar over the table header when 1 or more rows are selected. Requires `[selectionContext]` to be defined */\r\n  @Input() selectionToolbarTemplate?: TemplateRef<any>;\r\n  \r\n  /** Set to true if selectionContext is defined */\r\n  @HostBinding('class.is-selectable')\r\n  @Input() selectable = false;\r\n\r\n  /** Hides/shows the selection toolbar template when 1 or more rows is selected */\r\n  hasSelection = false\r\n\r\n  /**\r\n   * Enables form table styling to allow form controls inside table cells.\r\n   */\r\n  @HostBinding('class.is-form-table')\r\n  @Input() isForm = false;\r\n  \r\n  /**\r\n   * The current sort field and direction ('desc' or 'asc')\r\n   */\r\n  @Input() sort?: TableSort | TableSort[];\r\n\r\n  /**\r\n   * Emits the new sort value whenever a sortable column is clicked\r\n   */\r\n  @Output() sortChange: EventEmitter<TableSort | TableSort[]> = new EventEmitter();\r\n\r\n  /**\r\n   * True if a TableMasterRowComponent exists in the table content\r\n   */\r\n  @HostBinding('class.is-master-detail') hasMasterDetailRows = false;\r\n\r\n  @ContentChildren(TableMasterRowComponent, {descendants: true}) masterRows!: QueryList<TableMasterRowComponent>;\r\n\r\n  public resizableColumns!: QueryList<ResizableColumnComponent>;\r\n  @ContentChildren(ResizableColumnComponent, {descendants: true}) _resizableColumns!: QueryList<ResizableColumnComponent>;\r\n\r\n  @Input('resizableColumns') searchableTableResizableColumns?: QueryList<ResizableColumnComponent>;\r\n\r\n  /**\r\n   * Unlisten function returned by `renderer.listen()` that are called\r\n   * OnDestroy to clean up all listeners\r\n   */\r\n  private unlisteners: (() => void)[] = [];\r\n  \r\n  /** The current sort value */\r\n  private currentSort?: TableSort | TableSort[];\r\n  \r\n  /** A map of sortable `<th>` elements */\r\n  private sortableCols: {[field: string]: HTMLTableHeaderCellElement} = {};\r\n\r\n  private destroyed: Subject<void> = new Subject();\r\n  \r\n  constructor(private renderer: Renderer2, public el: ElementRef) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {\r\n      const newValue: TableSort | TableSort[] = changes.sort.currentValue;\r\n      if (!this.currentSort || !isEqual(newValue, this.currentSort)){\r\n        this.onSort(newValue);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngAfterContentInit(): void {\r\n    this.hasMasterDetailRows = !!this.masterRows.length;\r\n    \r\n    this.masterRows.changes.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(() => {\r\n      this.hasMasterDetailRows = !!this.masterRows.length;\r\n    });\r\n\r\n    this.resizableColumns = this.searchableTableResizableColumns || this._resizableColumns;\r\n  }\r\n  \r\n  ngAfterViewInit(): void {\r\n    if (this.sortable) {\r\n      this.initSortable();\r\n    }\r\n\r\n    if (this.selectable) {\r\n      this.initSelectable();\r\n    }\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    if (this.unlisteners.length) {\r\n      this.unlisteners.forEach(unlisten => unlisten());\r\n    }\r\n    \r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Get all elements inside the host element that match a selector \r\n   * @param selector - Any valid selector for [querySelectorAll](https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelectorAll)\r\n   */\r\n  private queryElements<T>(selector: string): T[] {\r\n    const table = this.el.nativeElement;\r\n    return Array.from(table.querySelectorAll(selector));\r\n  }\r\n\r\n  /**\r\n   * Store a `<th>` for each sortable column and attach a click\r\n   * event listener. \r\n   */\r\n  private initSortable(): void {\r\n    this.queryElements<HTMLTableHeaderCellElement>('th').forEach(th => {\r\n      if (th.hasAttribute('data-sortfield')) {\r\n        \r\n        const field = th.getAttribute('data-sortfield');\r\n        if (field) {\r\n          this.sortableCols[field] = th;\r\n          const unlisten = this.renderer.listen(th, 'click', (event: MouseEvent) => {\r\n\r\n            // Do not sort if the event target was the resizable handle\r\n            if (th.classList.contains('is-resizable') && event.target && (<HTMLElement>event.target).classList.contains('handle')) {\r\n              return;\r\n            }\r\n\r\n            let direction: TableSortDirection;\r\n\r\n            if (th.classList.contains('is-sorted-desc')) {\r\n              direction = 'asc';\r\n            } else {\r\n              direction = 'desc';\r\n            }\r\n\r\n            this.onSort({ field, direction });\r\n          });\r\n  \r\n          // Store unlisten function to remove listener when component is destroyed\r\n          this.unlisteners.push(unlisten);\r\n        } else {\r\n          console.error(`data-sortfield attribute was not assigned a value.`, th);\r\n        }\r\n      }\r\n    });\r\n\r\n    // Set the initial sort\r\n    if (this.sort) {\r\n      this.onSort(this.sort);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Find the sorted column and apply the desc or asc class in order\r\n   * to show the correct sort icon. If another field was sorted, reset it.\r\n   * Emit the current field and direction of the sorted column\r\n   * @param field \r\n   */\r\n  private onSort(sort: TableSort | TableSort[]): void {\r\n    this.clearSort();\r\n\r\n    if (Array.isArray(sort)) {\r\n      sort.forEach(s => this.applySort(s));\r\n    } else {\r\n      this.applySort(sort);\r\n    }\r\n\r\n    this.currentSort = cloneDeep(sort);\r\n    this.sortChange.emit(this.currentSort);\r\n  }\r\n\r\n  private clearSort() {\r\n    Object.values(this.sortableCols).forEach(th => {\r\n      this.renderer.removeClass(th, `is-sorted-asc`);\r\n      this.renderer.removeClass(th, `is-sorted-desc`);\r\n    });\r\n  }\r\n\r\n  private applySort(sort: TableSort) {\r\n    const th = this.sortableCols[sort.field];\r\n    if (!th) { return; }\r\n\r\n    this.renderer.addClass(th, `is-sorted-${sort.direction}`);\r\n  }\r\n\r\n  private initSelectable(): void {\r\n    this.selectionContext!.rowCheckboxes.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe((value: boolean[]) => {\r\n      this.hasSelection = value.some(v => v);\r\n    });\r\n  }\r\n}\r\n","<div class=\"selectable-table-toolbar\"\r\n     *ngIf=\"hasSelection && selectionToolbarTemplate\">\r\n  <ng-container *ngTemplateOutlet=\"selectionToolbarTemplate\"></ng-container>\r\n</div>\r\n<div id=\"{{id}}-scroll-container\" \r\n     class=\"table-scroll-container\">\r\n  <table class=\"main-table\"\r\n         id=\"{{id}}\"\r\n         [ecResizableTable]=\"resizable\"\r\n         [containerEl]=\"el\"\r\n         [sortableTable]=\"sortable\"\r\n         [resizableColumns]=\"resizableColumns\">\r\n    <ng-content></ng-content>\r\n  </table>\r\n</div>"]}
@@ -7145,7 +7145,7 @@ class TableComponent {
7145
7145
  if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {
7146
7146
  const newValue = changes.sort.currentValue;
7147
7147
  if (!this.currentSort || !isEqual(newValue, this.currentSort)) {
7148
- this.onSort(newValue.field, newValue.direction);
7148
+ this.onSort(newValue);
7149
7149
  }
7150
7150
  }
7151
7151
  }
@@ -7194,7 +7194,14 @@ class TableComponent {
7194
7194
  if (th.classList.contains('is-resizable') && event.target && event.target.classList.contains('handle')) {
7195
7195
  return;
7196
7196
  }
7197
- this.onSort(field);
7197
+ let direction;
7198
+ if (th.classList.contains('is-sorted-desc')) {
7199
+ direction = 'asc';
7200
+ }
7201
+ else {
7202
+ direction = 'desc';
7203
+ }
7204
+ this.onSort({ field, direction });
7198
7205
  });
7199
7206
  // Store unlisten function to remove listener when component is destroyed
7200
7207
  this.unlisteners.push(unlisten);
@@ -7206,7 +7213,7 @@ class TableComponent {
7206
7213
  });
7207
7214
  // Set the initial sort
7208
7215
  if (this.sort) {
7209
- this.onSort(this.sort.field, this.sort.direction);
7216
+ this.onSort(this.sort);
7210
7217
  }
7211
7218
  }
7212
7219
  /**
@@ -7215,32 +7222,29 @@ class TableComponent {
7215
7222
  * Emit the current field and direction of the sorted column
7216
7223
  * @param field
7217
7224
  */
7218
- onSort(field, direction) {
7219
- const th = this.sortableCols[field];
7220
- if (!th) {
7221
- return;
7222
- }
7223
- if (this.currentSort && this.currentSort.field !== field) {
7224
- this.renderer.removeClass(this.sortableCols[this.currentSort.field], `is-sorted-${this.currentSort.direction}`);
7225
- }
7226
- if (th.classList.contains('is-sorted-desc')) {
7227
- this.renderer.removeClass(th, 'is-sorted-desc');
7228
- if (!direction) {
7229
- direction = 'asc';
7230
- }
7225
+ onSort(sort) {
7226
+ this.clearSort();
7227
+ if (Array.isArray(sort)) {
7228
+ sort.forEach(s => this.applySort(s));
7231
7229
  }
7232
7230
  else {
7233
- if (th.classList.contains('is-sorted-asc')) {
7234
- this.renderer.removeClass(th, 'is-sorted-asc');
7235
- }
7236
- if (!direction) {
7237
- direction = 'desc';
7238
- }
7231
+ this.applySort(sort);
7232
+ }
7233
+ this.currentSort = cloneDeep(sort);
7234
+ this.sortChange.emit(this.currentSort);
7235
+ }
7236
+ clearSort() {
7237
+ Object.values(this.sortableCols).forEach(th => {
7238
+ this.renderer.removeClass(th, `is-sorted-asc`);
7239
+ this.renderer.removeClass(th, `is-sorted-desc`);
7240
+ });
7241
+ }
7242
+ applySort(sort) {
7243
+ const th = this.sortableCols[sort.field];
7244
+ if (!th) {
7245
+ return;
7239
7246
  }
7240
- this.renderer.addClass(th, `is-sorted-${direction}`);
7241
- const sortEvent = { field: field, direction: direction };
7242
- this.currentSort = sortEvent;
7243
- this.sortChange.emit(sortEvent);
7247
+ this.renderer.addClass(th, `is-sorted-${sort.direction}`);
7244
7248
  }
7245
7249
  initSelectable() {
7246
7250
  this.selectionContext.rowCheckboxes.valueChanges.pipe(takeUntil(this.destroyed)).subscribe((value) => {