@energycap/components 0.39.32 → 0.39.33-ECAP-26866-table-multi-sort-improvements.20241024-0933
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.
- package/esm2020/lib/display/table/table.component.mjs +32 -28
- package/fesm2015/energycap-components.mjs +30 -26
- package/fesm2015/energycap-components.mjs.map +1 -1
- package/fesm2020/energycap-components.mjs +30 -26
- package/fesm2020/energycap-components.mjs.map +1 -1
- package/lib/display/table/table.component.d.ts +4 -2
- package/package.json +1 -1
@@ -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
|
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
|
-
|
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
|
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(
|
151
|
-
|
152
|
-
if (
|
153
|
-
|
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
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
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":"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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 { 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
|
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
|
-
|
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
|
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(
|
7219
|
-
|
7220
|
-
if (
|
7221
|
-
|
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
|
-
|
7234
|
-
|
7235
|
-
|
7236
|
-
|
7237
|
-
|
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) => {
|