@tilde-nlp/ngx-common 2.0.16 → 2.0.18
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/filter-bar/filter-bar.component.mjs +78 -53
- package/esm2020/lib/multi-functional-table/multi-functional-table.component.mjs +3 -3
- package/fesm2015/tilde-nlp-ngx-common.mjs +103 -78
- package/fesm2015/tilde-nlp-ngx-common.mjs.map +1 -1
- package/fesm2020/tilde-nlp-ngx-common.mjs +100 -76
- package/fesm2020/tilde-nlp-ngx-common.mjs.map +1 -1
- package/lib/filter-bar/filter-bar.component.d.ts +25 -22
- package/package.json +1 -1
|
@@ -1,65 +1,67 @@
|
|
|
1
1
|
import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
|
-
import * as i1 from "@
|
|
5
|
-
import * as i2 from "@angular/
|
|
6
|
-
import * as i3 from "@angular/material/
|
|
7
|
-
import * as i4 from "@angular/
|
|
8
|
-
import * as i5 from "@angular/
|
|
9
|
-
import * as i6 from "@angular/
|
|
10
|
-
import * as i7 from "@angular/
|
|
11
|
-
import * as i8 from "@angular/material/
|
|
12
|
-
import * as i9 from "@angular/
|
|
13
|
-
import * as i10 from "@angular/material/
|
|
14
|
-
import * as i11 from "@angular/material/
|
|
15
|
-
import * as i12 from "@
|
|
4
|
+
import * as i1 from "@ngx-translate/core";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
import * as i3 from "@angular/material/form-field";
|
|
7
|
+
import * as i4 from "@angular/material/input";
|
|
8
|
+
import * as i5 from "@angular/flex-layout/flex";
|
|
9
|
+
import * as i6 from "@angular/flex-layout/extended";
|
|
10
|
+
import * as i7 from "@angular/material/select";
|
|
11
|
+
import * as i8 from "@angular/material/core";
|
|
12
|
+
import * as i9 from "@angular/forms";
|
|
13
|
+
import * as i10 from "@angular/material/tooltip";
|
|
14
|
+
import * as i11 from "@angular/material/button";
|
|
15
|
+
import * as i12 from "@angular/material/chips";
|
|
16
|
+
import * as i13 from "@angular/material/icon";
|
|
16
17
|
export class FilterBarComponent {
|
|
17
|
-
constructor(cdref) {
|
|
18
|
+
constructor(cdref, translate) {
|
|
18
19
|
this.cdref = cdref;
|
|
19
|
-
this.
|
|
20
|
-
this.filters = {
|
|
21
|
-
filters: {},
|
|
22
|
-
input: ""
|
|
23
|
-
};
|
|
24
|
-
this.filterOverflow = false;
|
|
20
|
+
this.translate = translate;
|
|
25
21
|
this.filterBarChange = new EventEmitter();
|
|
22
|
+
this._inputText = "";
|
|
23
|
+
/** To hide filters, but leave chips visible */
|
|
26
24
|
this.inputFormControlName = "input";
|
|
27
25
|
this.filterFormGroup = new FormGroup({});
|
|
28
26
|
this.form = new FormGroup({
|
|
29
27
|
[this.inputFormControlName]: new FormControl(""),
|
|
30
28
|
filters: this.filterFormGroup
|
|
31
29
|
});
|
|
32
|
-
this.
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
this.
|
|
40
|
-
this.emitFilters();
|
|
41
|
-
}
|
|
42
|
-
get showSearch() {
|
|
43
|
-
return !this.settings?.hideSearch;
|
|
30
|
+
this.filters = {
|
|
31
|
+
filters: {},
|
|
32
|
+
input: ""
|
|
33
|
+
};
|
|
34
|
+
this.filterOverflow = false;
|
|
35
|
+
this.activeFilterIndex = 0;
|
|
36
|
+
// when to show filter by name or id on mobile
|
|
37
|
+
this.searchIndex = -1;
|
|
44
38
|
}
|
|
45
|
-
|
|
46
|
-
|
|
39
|
+
onResize() {
|
|
40
|
+
this.checkOverflow();
|
|
47
41
|
}
|
|
48
42
|
set filterRowVisible(value) {
|
|
49
43
|
this._filterRowVisible = value;
|
|
50
44
|
this.cdref.detectChanges();
|
|
51
45
|
this.checkOverflow();
|
|
52
46
|
}
|
|
53
|
-
get filterRowVisible() { return this._filterRowVisible; }
|
|
54
|
-
onResize() {
|
|
55
|
-
this.checkOverflow();
|
|
56
|
-
}
|
|
57
47
|
// getter for cleaner template html
|
|
58
48
|
get filterFormGroupValue() {
|
|
59
49
|
return this.filterFormGroup.value;
|
|
60
50
|
}
|
|
51
|
+
get filterRowVisible() { return this._filterRowVisible; }
|
|
52
|
+
get inputText() {
|
|
53
|
+
return this._inputText;
|
|
54
|
+
}
|
|
55
|
+
get showSearch() {
|
|
56
|
+
return !this.settings?.hideSearch;
|
|
57
|
+
}
|
|
58
|
+
set inputText(value) {
|
|
59
|
+
this._inputText = value;
|
|
60
|
+
this.filters.input = this._inputText;
|
|
61
|
+
this.emitFilters();
|
|
62
|
+
}
|
|
61
63
|
ngOnInit() {
|
|
62
|
-
this.
|
|
64
|
+
this.getSortedFilters().forEach((field) => {
|
|
63
65
|
this.filterFormGroup.addControl(field.fieldName, new FormControl([]));
|
|
64
66
|
});
|
|
65
67
|
this.subscribeToFormValueChanges();
|
|
@@ -79,11 +81,17 @@ export class FilterBarComponent {
|
|
|
79
81
|
emitFilters() {
|
|
80
82
|
this.filterBarChange.next(this.filters);
|
|
81
83
|
}
|
|
82
|
-
|
|
83
|
-
this.
|
|
84
|
+
switchRight() {
|
|
85
|
+
if (this.settings?.filters && this.settings?.filters.length === this.activeFilterIndex + 1) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
this.activeFilterIndex += 1;
|
|
84
89
|
}
|
|
85
|
-
|
|
86
|
-
|
|
90
|
+
switchLeft() {
|
|
91
|
+
if (this.activeFilterIndex === this.searchIndex) {
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
this.activeFilterIndex -= 1;
|
|
87
95
|
}
|
|
88
96
|
removeFilter(key, ix) {
|
|
89
97
|
const control = this.filterFormGroup.get(key);
|
|
@@ -95,6 +103,23 @@ export class FilterBarComponent {
|
|
|
95
103
|
control.setValue(newArray);
|
|
96
104
|
// this.form.setValue(this.filters.filters);
|
|
97
105
|
}
|
|
106
|
+
getSortedFilters() {
|
|
107
|
+
if (this.settings?.filters) {
|
|
108
|
+
this.settings.filters.forEach((item) => {
|
|
109
|
+
item.values.sort((a, b) => {
|
|
110
|
+
const aTranslated = this.translate.instant(a.key).toLowerCase();
|
|
111
|
+
const bTranslated = this.translate.instant(b.key).toLowerCase();
|
|
112
|
+
if (aTranslated < bTranslated)
|
|
113
|
+
return -1;
|
|
114
|
+
if (aTranslated > bTranslated)
|
|
115
|
+
return 1;
|
|
116
|
+
return 0;
|
|
117
|
+
});
|
|
118
|
+
});
|
|
119
|
+
return this.settings.filters;
|
|
120
|
+
}
|
|
121
|
+
return [];
|
|
122
|
+
}
|
|
98
123
|
checkOverflow() {
|
|
99
124
|
if (this.filterWrapper) {
|
|
100
125
|
this.filterOverflow = this.filterWrapper.nativeElement.clientWidth < this.filterWrapper.nativeElement.scrollWidth;
|
|
@@ -115,22 +140,22 @@ export class FilterBarComponent {
|
|
|
115
140
|
});
|
|
116
141
|
}
|
|
117
142
|
}
|
|
118
|
-
FilterBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterBarComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
119
|
-
FilterBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: FilterBarComponent, selector: "tld-filter-bar", inputs: { settings: "settings", filterRowVisible: "filterRowVisible" }, outputs: { filterBarChange: "filterBarChange" }, host: { listeners: { "window:resize": "onResize()" } }, viewQueries: [{ propertyName: "filterWrapper", first: true, predicate: ["filterWrapper"], descendants: true }], ngImport: i0, template: "<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n <
|
|
143
|
+
FilterBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterBarComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.TranslateService }], target: i0.ɵɵFactoryTarget.Component });
|
|
144
|
+
FilterBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: FilterBarComponent, selector: "tld-filter-bar", inputs: { settings: "settings", filterRowVisible: "filterRowVisible" }, outputs: { filterBarChange: "filterBarChange" }, host: { listeners: { "window:resize": "onResize()" } }, viewQueries: [{ propertyName: "filterWrapper", first: true, predicate: ["filterWrapper"], descendants: true }], ngImport: i0, template: "<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n <div fxLayout=\"row\" class=\"filter-wrapper\" #filterWrapper [formGroup]=\"form\">\r\n <mat-form-field fxFlex *ngIf=\"showSearch && !filterOverflow || activeFilterIndex === searchIndex\" class=\"filter-bar-search-input\">\r\n <span matPrefix class=\"material-icons-outlined\">\r\n {{prefixIcon}}\r\n </span>\r\n <span matSuffix class=\"material-icons-outlined\" *ngIf=\"showSuffixIcon\" [matTooltip]=\"searchTooltip | translate\">\r\n {{suffixIcon}}\r\n </span>\r\n <input class=\"search-input\" matInput [placeholder]=\"searchTitle | translate\"\r\n [formControlName]=\"inputFormControlName\">\r\n </mat-form-field>\r\n <ng-container [formGroup]=\"filterFormGroup\">\r\n\r\n <mat-form-field [ngClass]=\"{'hidden': filterOverflow && activeFilterIndex !== i}\" *ngFor=\"let filter of settings?.filters; let i = index\">\r\n <mat-label>{{filter.title | translate}}</mat-label>\r\n <mat-select multiple [formControlName]=\"filter.fieldName\">\r\n <mat-option *ngFor=\"let value of filter.values\" [value]=\"value\">\r\n {{value.key | translate : {default: value.value | titlecase } }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </ng-container>\r\n </div>\r\n <button mat-icon-button (click)=\"switchLeft()\" *ngIf=\"filterOverflow\">\r\n <span class=\"material-icons-outlined\">\r\n arrow_back_ios\r\n </span>\r\n </button>\r\n <button mat-icon-button *ngIf=\"filterOverflow\" (click)=\"switchRight()\">\r\n <span class=\"material-icons-outlined\">\r\n arrow_forward_ios\r\n </span>\r\n </button>\r\n</div>\r\n<mat-chip-list>\r\n <div *ngFor=\"let filter of settings?.filters\">\r\n <mat-chip *ngFor=\"let filterValue of filterFormGroupValue[filter.fieldName]; let ix=index\"\r\n (removed)=\"removeFilter(filter.fieldName, ix)\" class=\"text-s\">\r\n <button matChipRemove>\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n <span class=\"chip-filter-title\">{{filter.title | translate}}:</span>\r\n <span class=\"chip-value semi-bold\"> {{filterValue.key | translate : {default: filterValue.value | titlecase }\r\n }}</span>\r\n </mat-chip>\r\n </div>\r\n</mat-chip-list>\r\n", styles: [":host{display:inline-block}.filter-row{display:flex;background-color:var(--base-95);min-height:40px;min-width:100%;border-radius:.5rem;padding:.5rem 1.25rem 1rem;max-width:100%;align-items:baseline;margin-bottom:.5rem}.filter-row ::ng-deep .mat-form-field-wrapper{padding-bottom:0!important}.filter-row ::ng-deep .mat-form-field-underline{bottom:0!important}.search-input{background-color:unset;border:none}mat-form-field+mat-form-field{margin-left:1rem}.filter-bar-search-input{width:100%}.filter-bar-search-input::ng-deep .material-icons-outlined{vertical-align:bottom}.material-icons-outlined{color:var(--base-40)}.filter-wrapper{max-width:100%;overflow-x:hidden;overflow-y:hidden;display:flex;flex:1}.mat-chip.mat-standard-chip{background-color:var(--base-95);padding-left:.5em;padding-right:.75em;border:.5px solid var(--base-70)}.mat-chip.mat-standard-chip span{display:inline-block}.mat-chip.mat-standard-chip .chip-filter-title+.chip-value{margin-left:.25rem}.mat-chip .mat-chip-remove{margin-left:0;margin-right:.5em;color:var(--base-40);opacity:1}.hidden{display:none}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "directive", type: i3.MatPrefix, selector: "[matPrefix]" }, { kind: "directive", type: i3.MatSuffix, selector: "[matSuffix]" }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "directive", type: i5.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i5.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { kind: "directive", type: i6.DefaultClassDirective, selector: " [ngClass], [ngClass.xs], [ngClass.sm], [ngClass.md], [ngClass.lg], [ngClass.xl], [ngClass.lt-sm], [ngClass.lt-md], [ngClass.lt-lg], [ngClass.lt-xl], [ngClass.gt-xs], [ngClass.gt-sm], [ngClass.gt-md], [ngClass.gt-lg]", inputs: ["ngClass", "ngClass.xs", "ngClass.sm", "ngClass.md", "ngClass.lg", "ngClass.xl", "ngClass.lt-sm", "ngClass.lt-md", "ngClass.lt-lg", "ngClass.lt-xl", "ngClass.gt-xs", "ngClass.gt-sm", "ngClass.gt-md", "ngClass.gt-lg"] }, { kind: "component", type: i7.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex"], exportAs: ["matSelect"] }, { kind: "component", type: i8.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i9.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i9.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i11.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i12.MatChipList, selector: "mat-chip-list", inputs: ["role", "aria-describedby", "errorStateMatcher", "multiple", "compareWith", "value", "required", "placeholder", "disabled", "aria-orientation", "selectable", "tabIndex"], outputs: ["change", "valueChange"], exportAs: ["matChipList"] }, { kind: "directive", type: i12.MatChip, selector: "mat-basic-chip, [mat-basic-chip], mat-chip, [mat-chip]", inputs: ["color", "disableRipple", "tabIndex", "role", "selected", "value", "selectable", "disabled", "removable"], outputs: ["selectionChange", "destroyed", "removed"], exportAs: ["matChip"] }, { kind: "directive", type: i12.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i13.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i9.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i9.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "pipe", type: i2.TitleCasePipe, name: "titlecase" }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }] });
|
|
120
145
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterBarComponent, decorators: [{
|
|
121
146
|
type: Component,
|
|
122
|
-
args: [{ selector: 'tld-filter-bar', template: "<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n <
|
|
123
|
-
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: {
|
|
124
|
-
type: Output
|
|
125
|
-
}], settings: [{
|
|
126
|
-
type: Input
|
|
127
|
-
}], filterRowVisible: [{
|
|
128
|
-
type: Input
|
|
129
|
-
}], filterWrapper: [{
|
|
147
|
+
args: [{ selector: 'tld-filter-bar', template: "<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n <div fxLayout=\"row\" class=\"filter-wrapper\" #filterWrapper [formGroup]=\"form\">\r\n <mat-form-field fxFlex *ngIf=\"showSearch && !filterOverflow || activeFilterIndex === searchIndex\" class=\"filter-bar-search-input\">\r\n <span matPrefix class=\"material-icons-outlined\">\r\n {{prefixIcon}}\r\n </span>\r\n <span matSuffix class=\"material-icons-outlined\" *ngIf=\"showSuffixIcon\" [matTooltip]=\"searchTooltip | translate\">\r\n {{suffixIcon}}\r\n </span>\r\n <input class=\"search-input\" matInput [placeholder]=\"searchTitle | translate\"\r\n [formControlName]=\"inputFormControlName\">\r\n </mat-form-field>\r\n <ng-container [formGroup]=\"filterFormGroup\">\r\n\r\n <mat-form-field [ngClass]=\"{'hidden': filterOverflow && activeFilterIndex !== i}\" *ngFor=\"let filter of settings?.filters; let i = index\">\r\n <mat-label>{{filter.title | translate}}</mat-label>\r\n <mat-select multiple [formControlName]=\"filter.fieldName\">\r\n <mat-option *ngFor=\"let value of filter.values\" [value]=\"value\">\r\n {{value.key | translate : {default: value.value | titlecase } }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </ng-container>\r\n </div>\r\n <button mat-icon-button (click)=\"switchLeft()\" *ngIf=\"filterOverflow\">\r\n <span class=\"material-icons-outlined\">\r\n arrow_back_ios\r\n </span>\r\n </button>\r\n <button mat-icon-button *ngIf=\"filterOverflow\" (click)=\"switchRight()\">\r\n <span class=\"material-icons-outlined\">\r\n arrow_forward_ios\r\n </span>\r\n </button>\r\n</div>\r\n<mat-chip-list>\r\n <div *ngFor=\"let filter of settings?.filters\">\r\n <mat-chip *ngFor=\"let filterValue of filterFormGroupValue[filter.fieldName]; let ix=index\"\r\n (removed)=\"removeFilter(filter.fieldName, ix)\" class=\"text-s\">\r\n <button matChipRemove>\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n <span class=\"chip-filter-title\">{{filter.title | translate}}:</span>\r\n <span class=\"chip-value semi-bold\"> {{filterValue.key | translate : {default: filterValue.value | titlecase }\r\n }}</span>\r\n </mat-chip>\r\n </div>\r\n</mat-chip-list>\r\n", styles: [":host{display:inline-block}.filter-row{display:flex;background-color:var(--base-95);min-height:40px;min-width:100%;border-radius:.5rem;padding:.5rem 1.25rem 1rem;max-width:100%;align-items:baseline;margin-bottom:.5rem}.filter-row ::ng-deep .mat-form-field-wrapper{padding-bottom:0!important}.filter-row ::ng-deep .mat-form-field-underline{bottom:0!important}.search-input{background-color:unset;border:none}mat-form-field+mat-form-field{margin-left:1rem}.filter-bar-search-input{width:100%}.filter-bar-search-input::ng-deep .material-icons-outlined{vertical-align:bottom}.material-icons-outlined{color:var(--base-40)}.filter-wrapper{max-width:100%;overflow-x:hidden;overflow-y:hidden;display:flex;flex:1}.mat-chip.mat-standard-chip{background-color:var(--base-95);padding-left:.5em;padding-right:.75em;border:.5px solid var(--base-70)}.mat-chip.mat-standard-chip span{display:inline-block}.mat-chip.mat-standard-chip .chip-filter-title+.chip-value{margin-left:.25rem}.mat-chip .mat-chip-remove{margin-left:0;margin-right:.5em;color:var(--base-40);opacity:1}.hidden{display:none}\n"] }]
|
|
148
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.TranslateService }]; }, propDecorators: { filterWrapper: [{
|
|
130
149
|
type: ViewChild,
|
|
131
150
|
args: ["filterWrapper"]
|
|
132
151
|
}], onResize: [{
|
|
133
152
|
type: HostListener,
|
|
134
153
|
args: ['window:resize', []]
|
|
154
|
+
}], settings: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], filterRowVisible: [{
|
|
157
|
+
type: Input
|
|
158
|
+
}], filterBarChange: [{
|
|
159
|
+
type: Output
|
|
135
160
|
}] } });
|
|
136
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-bar.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-common/src/lib/filter-bar/filter-bar.component.ts","../../../../../projects/ngx-common/src/lib/filter-bar/filter-bar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, OnDestroy, OnInit, Output, ViewChild } from '@angular/core';\r\nimport { FormControl, FormGroup } from '@angular/forms';\r\nimport { Subscription } from 'rxjs';\r\nimport { FilterBarFilterItem } from './models';\r\nimport { FilterBarChangeEvent } from './models/filter-bar-change-event.model';\r\nimport { FilterBarSettings } from './models/filter-bar-settings.model';\r\n\r\n@Component({\r\n  // eslint-disable-next-line @angular-eslint/component-selector\r\n  selector: 'tld-filter-bar',\r\n  templateUrl: './filter-bar.component.html',\r\n  styleUrls: ['./filter-bar.component.scss']\r\n})\r\nexport class FilterBarComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  private _inputText = \"\";\r\n  get inputText() {\r\n    return this._inputText;\r\n  }\r\n  set inputText(value) {\r\n    this._inputText = value;\r\n    this.filters.input = this._inputText;\r\n    this.emitFilters();\r\n  }\r\n\r\n  searchTitle!: string;\r\n  searchTooltip!: string;\r\n  suffixIcon!: string;\r\n  prefixIcon!: string;\r\n  showSuffixIcon!: boolean;\r\n\r\n  filters: FilterBarChangeEvent = {\r\n    filters: {},\r\n    input: \"\"\r\n  };\r\n\r\n\r\n  filterOverflow = false;\r\n  @Output() filterBarChange: EventEmitter<FilterBarChangeEvent> = new EventEmitter<FilterBarChangeEvent>();\r\n\r\n  get showSearch() {\r\n    return !this.settings?.hideSearch;\r\n  }\r\n\r\n  get fields() {\r\n    return this.settings?.filters ?? [];\r\n  }\r\n  @Input() settings!: FilterBarSettings;\r\n\r\n  /** To hide filters, but leave chips visible */\r\n  private _filterRowVisible!: boolean;\r\n  @Input() set filterRowVisible(value: boolean) {\r\n    this._filterRowVisible = value;\r\n    this.cdref.detectChanges();\r\n    this.checkOverflow();\r\n  }\r\n  get filterRowVisible() { return this._filterRowVisible; }\r\n\r\n  @ViewChild(\"filterWrapper\") filterWrapper!: ElementRef;\r\n  @HostListener('window:resize', [])\r\n  onResize() {\r\n    this.checkOverflow();\r\n  }\r\n  private formChangesSubscription!: Subscription;\r\n\r\n  readonly inputFormControlName = \"input\";\r\n  readonly filterFormGroup = new FormGroup({});\r\n  readonly form = new FormGroup({\r\n    [this.inputFormControlName]: new FormControl(\"\"),\r\n    filters: this.filterFormGroup\r\n  });\r\n  // getter for cleaner template html\r\n  get filterFormGroupValue(): { [key: string]: FilterBarFilterItem[] } {\r\n    return this.filterFormGroup.value;\r\n  }\r\n\r\n\r\n  private readonly defaultScrollWidth = 180;\r\n\r\n  constructor(private cdref: ChangeDetectorRef) { }\r\n\r\n  ngOnInit() {\r\n\r\n    this.fields.forEach((field) => {\r\n      this.filterFormGroup.addControl(field.fieldName, new FormControl([]));\r\n    })\r\n\r\n    this.subscribeToFormValueChanges();\r\n    this.searchTooltip = this.settings?.searchTooltip ?? 'FILTER_BAR.SEARCH_TOOLTIP';\r\n    this.searchTitle = this.settings?.searchTitle ?? 'FILTER_BAR.SEARCH';\r\n    this.prefixIcon = this.settings?.prefixIcon ?? \"filter_alt\";\r\n    this.suffixIcon = this.settings?.suffixIcon ?? \"question_mark\";\r\n    this.showSuffixIcon = this.settings?.showSuffixIcon ?? true;\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.checkOverflow();\r\n    this.cdref.detectChanges();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.formChangesSubscription.unsubscribe();\r\n  }\r\n\r\n  emitFilters() {\r\n    this.filterBarChange.next(this.filters);\r\n  }\r\n\r\n  scrollRight() {\r\n    this.filterWrapper.nativeElement.scrollTo({ left: (this.filterWrapper.nativeElement.scrollLeft + this.defaultScrollWidth), behavior: 'smooth' });\r\n  }\r\n\r\n  scrollLeft() {\r\n    this.filterWrapper.nativeElement.scrollTo({ left: (this.filterWrapper.nativeElement.scrollLeft - this.defaultScrollWidth), behavior: 'smooth' });\r\n  }\r\n\r\n  removeFilter(key: string, ix: number) {\r\n\r\n    const control = this.filterFormGroup.get(key);\r\n    if (!control) {\r\n      return;\r\n    }\r\n\r\n    const newArray = control.value as [];\r\n    newArray.splice(ix, 1);\r\n    control.setValue(newArray);\r\n    // this.form.setValue(this.filters.filters);\r\n  }\r\n\r\n  private checkOverflow() {\r\n    if (this.filterWrapper) {\r\n      this.filterOverflow = this.filterWrapper.nativeElement.clientWidth < this.filterWrapper.nativeElement.scrollWidth;\r\n    }\r\n  }\r\n\r\n  private subscribeToFormValueChanges() {\r\n    this.formChangesSubscription = this.form.valueChanges.subscribe(() => {\r\n      this.filters = {\r\n        input: this.form.value.input ?? \"\",\r\n        filters: {}\r\n      };\r\n\r\n      for (const filterKey in this.form.value.filters) {\r\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n        const filter: FilterBarFilterItem[] = (this.form.value.filters as any)[filterKey] as FilterBarFilterItem[];\r\n        this.filters.filters[filterKey] = filter.map((item) => item.value)\r\n      }\r\n      this.emitFilters();\r\n    })\r\n  }\r\n}\r\n","<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n  <button mat-icon-button (click)=\"scrollLeft()\" *ngIf=\"filterOverflow\">\r\n    <span class=\"material-icons-outlined\">\r\n      arrow_back_ios\r\n    </span>\r\n  </button>\r\n  <div fxLayout=\"row\" class=\"filter-wrapper\" #filterWrapper [formGroup]=\"form\">\r\n    <mat-form-field fxFlex *ngIf=\"showSearch\" class=\"filter-bar-search-input\">\r\n      <span matPrefix class=\"material-icons-outlined\">\r\n        {{prefixIcon}}\r\n      </span>\r\n      <span matSuffix class=\"material-icons-outlined\" *ngIf=\"showSuffixIcon\" [matTooltip]=\"searchTooltip | translate\">\r\n        {{suffixIcon}}\r\n      </span>\r\n      <input class=\"search-input\" matInput [placeholder]=\"searchTitle | translate\"\r\n        [formControlName]=\"inputFormControlName\">\r\n    </mat-form-field>\r\n    <ng-container [formGroup]=\"filterFormGroup\">\r\n\r\n      <mat-form-field *ngFor=\"let filter of fields\">\r\n        <mat-label>{{filter.title | translate}}</mat-label>\r\n        <mat-select multiple [formControlName]=\"filter.fieldName\">\r\n          <mat-option *ngFor=\"let value of filter.values\" [value]=\"value\">\r\n            {{value.key | translate : {default: value.value | titlecase } }}\r\n          </mat-option>\r\n        </mat-select>\r\n      </mat-form-field>\r\n    </ng-container>\r\n  </div>\r\n  <button mat-icon-button *ngIf=\"filterOverflow\" (click)=\"scrollRight()\">\r\n    <span class=\"material-icons-outlined\">\r\n      arrow_forward_ios\r\n    </span>\r\n  </button>\r\n</div>\r\n<mat-chip-list>\r\n  <div *ngFor=\"let filter of fields\">\r\n    <mat-chip *ngFor=\"let filterValue of filterFormGroupValue[filter.fieldName]; let ix=index\"\r\n      (removed)=\"removeFilter(filter.fieldName, ix)\" class=\"text-s\">\r\n      <button matChipRemove>\r\n        <mat-icon>close</mat-icon>\r\n      </button>\r\n      <span class=\"chip-filter-title\">{{filter.title | translate}}:</span>\r\n      <span class=\"chip-value semi-bold\"> {{filterValue.key | translate : {default: filterValue.value | titlecase }\r\n        }}</span>\r\n    </mat-chip>\r\n  </div>\r\n</mat-chip-list>\r\n"]}
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-bar.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-common/src/lib/filter-bar/filter-bar.component.ts","../../../../../projects/ngx-common/src/lib/filter-bar/filter-bar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, OnDestroy, OnInit, Output, ViewChild } from '@angular/core';\r\nimport { FormControl, FormGroup } from '@angular/forms';\r\nimport { Subscription } from 'rxjs';\r\nimport { FilterBarFilterItem } from './models';\r\nimport { FilterBarChangeEvent } from './models/filter-bar-change-event.model';\r\nimport { FilterBarSettings } from './models/filter-bar-settings.model';\r\nimport { TranslateService } from '@ngx-translate/core';\r\n\r\n@Component({\r\n  // eslint-disable-next-line @angular-eslint/component-selector\r\n  selector: 'tld-filter-bar',\r\n  templateUrl: './filter-bar.component.html',\r\n  styleUrls: ['./filter-bar.component.scss']\r\n})\r\nexport class FilterBarComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  @ViewChild(\"filterWrapper\") filterWrapper!: ElementRef;\r\n  @HostListener('window:resize', [])\r\n  onResize() {\r\n    this.checkOverflow();\r\n  }\r\n\r\n  @Input() settings!: FilterBarSettings;\r\n  @Input() set filterRowVisible(value: boolean) {\r\n    this._filterRowVisible = value;\r\n    this.cdref.detectChanges();\r\n    this.checkOverflow();\r\n  }\r\n\r\n  @Output() filterBarChange: EventEmitter<FilterBarChangeEvent> = new EventEmitter<FilterBarChangeEvent>();\r\n\r\n  private _inputText = \"\";\r\n  private _filterRowVisible!: boolean;\r\n  private formChangesSubscription!: Subscription;\r\n\r\n  /** To hide filters, but leave chips visible */\r\n  readonly inputFormControlName = \"input\";\r\n  readonly filterFormGroup = new FormGroup({});\r\n  readonly form = new FormGroup({\r\n    [this.inputFormControlName]: new FormControl(\"\"),\r\n    filters: this.filterFormGroup\r\n  });\r\n\r\n  searchTitle!: string;\r\n  searchTooltip!: string;\r\n  suffixIcon!: string;\r\n  prefixIcon!: string;\r\n  showSuffixIcon!: boolean;\r\n  filters: FilterBarChangeEvent = {\r\n    filters: {},\r\n    input: \"\"\r\n  };\r\n\r\n  filterOverflow = false;\r\n  activeFilterIndex = 0;\r\n\r\n  // when to show filter by name or id on mobile\r\n  searchIndex = -1;\r\n\r\n   // getter for cleaner template html\r\n  get filterFormGroupValue(): { [key: string]: FilterBarFilterItem[] } {\r\n    return this.filterFormGroup.value;\r\n  }\r\n  get filterRowVisible() { return this._filterRowVisible; }\r\n  get inputText() {\r\n    return this._inputText;\r\n  }\r\n  get showSearch() {\r\n    return !this.settings?.hideSearch;\r\n  }\r\n\r\n  set inputText(value) {\r\n    this._inputText = value;\r\n    this.filters.input = this._inputText;\r\n    this.emitFilters();\r\n  }\r\n\r\n  constructor(private cdref: ChangeDetectorRef, private translate: TranslateService) { }\r\n\r\n  ngOnInit() {\r\n    this.getSortedFilters().forEach((field) => {\r\n      this.filterFormGroup.addControl(field.fieldName, new FormControl([]));\r\n    })\r\n\r\n    this.subscribeToFormValueChanges();\r\n    this.searchTooltip = this.settings?.searchTooltip ?? 'FILTER_BAR.SEARCH_TOOLTIP';\r\n    this.searchTitle = this.settings?.searchTitle ?? 'FILTER_BAR.SEARCH';\r\n    this.prefixIcon = this.settings?.prefixIcon ?? \"filter_alt\";\r\n    this.suffixIcon = this.settings?.suffixIcon ?? \"question_mark\";\r\n    this.showSuffixIcon = this.settings?.showSuffixIcon ?? true;\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.checkOverflow();\r\n    this.cdref.detectChanges();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.formChangesSubscription.unsubscribe();\r\n  }\r\n\r\n  emitFilters() {\r\n    this.filterBarChange.next(this.filters);\r\n  }\r\n\r\n  switchRight() {\r\n    if (this.settings?.filters && this.settings?.filters.length === this.activeFilterIndex + 1) {\r\n      return;\r\n    }\r\n    this.activeFilterIndex += 1;\r\n  }\r\n\r\n  switchLeft() {\r\n    if (this.activeFilterIndex === this.searchIndex) {\r\n      return;\r\n    }\r\n    this.activeFilterIndex -= 1;\r\n  }\r\n\r\n  removeFilter(key: string, ix: number) {\r\n    const control = this.filterFormGroup.get(key);\r\n    \r\n    if (!control) {\r\n      return;\r\n    }\r\n\r\n    const newArray = control.value as [];\r\n    newArray.splice(ix, 1);\r\n    control.setValue(newArray);\r\n    // this.form.setValue(this.filters.filters);\r\n  }\r\n\r\n  private getSortedFilters() {\r\n    if (this.settings?.filters) {\r\n      this.settings.filters.forEach((item) => {\r\n        item.values.sort((a, b) => {\r\n          const aTranslated = this.translate.instant(a.key).toLowerCase();\r\n          const bTranslated = this.translate.instant(b.key).toLowerCase();\r\n          if (aTranslated < bTranslated) return -1;\r\n          if (aTranslated > bTranslated) return 1;\r\n          return 0;\r\n        });\r\n      });\r\n      return this.settings.filters;\r\n    }\r\n  \r\n    return [];\r\n  }\r\n\r\n  private checkOverflow() {\r\n    if (this.filterWrapper) {\r\n      this.filterOverflow = this.filterWrapper.nativeElement.clientWidth < this.filterWrapper.nativeElement.scrollWidth;\r\n    }\r\n  }\r\n\r\n  private subscribeToFormValueChanges() {\r\n    this.formChangesSubscription = this.form.valueChanges.subscribe(() => {\r\n      this.filters = {\r\n        input: this.form.value.input ?? \"\",\r\n        filters: {}\r\n      };\r\n\r\n      for (const filterKey in this.form.value.filters) {\r\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n        const filter: FilterBarFilterItem[] = (this.form.value.filters as any)[filterKey] as FilterBarFilterItem[];\r\n        this.filters.filters[filterKey] = filter.map((item) => item.value)\r\n      }\r\n      this.emitFilters();\r\n    })\r\n  }\r\n}\r\n","<div class=\"filter-row\" *ngIf=\"filterRowVisible\">\r\n  <div fxLayout=\"row\" class=\"filter-wrapper\" #filterWrapper [formGroup]=\"form\">\r\n    <mat-form-field fxFlex *ngIf=\"showSearch && !filterOverflow || activeFilterIndex === searchIndex\" class=\"filter-bar-search-input\">\r\n      <span matPrefix class=\"material-icons-outlined\">\r\n        {{prefixIcon}}\r\n      </span>\r\n      <span matSuffix class=\"material-icons-outlined\" *ngIf=\"showSuffixIcon\" [matTooltip]=\"searchTooltip | translate\">\r\n        {{suffixIcon}}\r\n      </span>\r\n      <input class=\"search-input\" matInput [placeholder]=\"searchTitle | translate\"\r\n        [formControlName]=\"inputFormControlName\">\r\n    </mat-form-field>\r\n    <ng-container [formGroup]=\"filterFormGroup\">\r\n\r\n      <mat-form-field [ngClass]=\"{'hidden': filterOverflow && activeFilterIndex !== i}\" *ngFor=\"let filter of settings?.filters; let i = index\">\r\n        <mat-label>{{filter.title | translate}}</mat-label>\r\n        <mat-select multiple [formControlName]=\"filter.fieldName\">\r\n          <mat-option *ngFor=\"let value of filter.values\" [value]=\"value\">\r\n            {{value.key | translate : {default: value.value | titlecase } }}\r\n          </mat-option>\r\n        </mat-select>\r\n      </mat-form-field>\r\n    </ng-container>\r\n  </div>\r\n  <button mat-icon-button (click)=\"switchLeft()\" *ngIf=\"filterOverflow\">\r\n    <span class=\"material-icons-outlined\">\r\n      arrow_back_ios\r\n    </span>\r\n  </button>\r\n  <button mat-icon-button *ngIf=\"filterOverflow\" (click)=\"switchRight()\">\r\n    <span class=\"material-icons-outlined\">\r\n      arrow_forward_ios\r\n    </span>\r\n  </button>\r\n</div>\r\n<mat-chip-list>\r\n  <div *ngFor=\"let filter of settings?.filters\">\r\n    <mat-chip *ngFor=\"let filterValue of filterFormGroupValue[filter.fieldName]; let ix=index\"\r\n      (removed)=\"removeFilter(filter.fieldName, ix)\" class=\"text-s\">\r\n      <button matChipRemove>\r\n        <mat-icon>close</mat-icon>\r\n      </button>\r\n      <span class=\"chip-filter-title\">{{filter.title | translate}}:</span>\r\n      <span class=\"chip-value semi-bold\"> {{filterValue.key | translate : {default: filterValue.value | titlecase }\r\n        }}</span>\r\n    </mat-chip>\r\n  </div>\r\n</mat-chip-list>\r\n"]}
|
|
@@ -188,10 +188,10 @@ export class MultiFunctionalTableComponent {
|
|
|
188
188
|
}
|
|
189
189
|
}
|
|
190
190
|
MultiFunctionalTableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MultiFunctionalTableComponent, deps: [{ token: i1.DOMService }, { token: i2.TranslateService }], target: i0.ɵɵFactoryTarget.Component });
|
|
191
|
-
MultiFunctionalTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MultiFunctionalTableComponent, selector: "tld-multi-functional-table", inputs: { config: "config", highlightedElements: "highlightedElements", selection: "selection" }, outputs: { filterBarChange: "filterBarChange", exported: "exported", selectionChange: "selectionChange" }, queries: [{ propertyName: "noDataRow", first: true, predicate: MatNoDataRow, descendants: true }, { propertyName: "headerRowDefs", predicate: MatHeaderRowDef }, { propertyName: "rowDefs", predicate: MatRowDef, descendants: true }, { propertyName: "columnDefs", predicate: MatColumnDef }], viewQueries: [{ propertyName: "table", first: true, predicate: MatTable, descendants: true, static: true }, { propertyName: "sort", first: true, predicate: MatSort, descendants: true }, { propertyName: "tableElementRef", first: true, predicate: MatTable, descendants: true, read: ElementRef }], ngImport: i0, template: "<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n <div fxLayout=\"row\">\r\n <div fxFlex fxLayoutGap=\"1rem\">\r\n <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n <span class=\"material-icons column-select-icon\">menu</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n </button>\r\n\r\n <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n <span class=\"material-icons\">filter_list</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n </button>\r\n\r\n <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n <span class=\"material-icons-outlined\">cloud_download</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n </button>\r\n </div>\r\n\r\n <ng-content select=\"[additionalActions]\"></ng-content>\r\n </div>\r\n\r\n <mat-menu #columnMenu=\"matMenu\">\r\n <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n <div *ngFor=\"let column of configurableColumns\">\r\n <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n {{column.displayName | translate}}\r\n </mat-checkbox>\r\n </div>\r\n </div>\r\n </mat-menu>\r\n\r\n <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n (filterBarChange)=\"filtersChanged($event)\">\r\n </tld-filter-bar>\r\n <table #table mat-table [dataSource]=\"config.dataSource\">\r\n <ng-content></ng-content>\r\n <ng-container [matColumnDef]=\"batchColumnName\">\r\n <th class=\"row-select\" mat-header-cell *matHeaderCellDef disable-export>\r\n <mat-checkbox [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (change)=\"toggleAllRowSelection()\"\r\n [checked]=\"matSelection.hasValue() && isAllSelected()\"\r\n [indeterminate]=\"matSelection.hasValue() && !isAllSelected()\">\r\n </mat-checkbox>\r\n </th>\r\n <td class=\"row-select\" mat-cell *matCellDef=\"let element\" disable-export>\r\n <mat-checkbox *ngIf=\"hoveredRow === element || matSelection.isSelected(element)\" [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (click)=\"$event.stopPropagation()\"\r\n (change)=\"toggleElementSelection(element)\" [checked]=\"matSelection.isSelected(element)\">\r\n </mat-checkbox>\r\n </td>\r\n </ng-container>\r\n <tr mat-header-row *matHeaderRowDef=\"displayColumns\" sticky></tr>\r\n <tr mat-row *matRowDef=\"let row; columns: displayColumns\" (mouseover)=\"hoveredRow = row\" (mouseleave)=\"hoveredRow = null\" [class.highlight]=\"highlightElement(row)\"></tr>\r\n\r\n <ng-container *ngIf=\"noDataRowActive\">\r\n <tr *matNoDataRow>\r\n <!-- add random number to make sure it takes full width -->\r\n <td colspan=\"99\">\r\n <div class=\"no-engines-wrapper\">\r\n <ng-container *ngIf=\"!noDataRowConfig.loading; else loading\">\r\n <div>\r\n <span class=\"material-icons-outlined\">\r\n {{noDataRowIcon}}\r\n </span>\r\n </div>\r\n <div class=\"text-xl-semi-bold\" *ngIf=\"noDataRowConfig.title\"\r\n [innerHtml]=\"noDataRowConfig.title | translate: noDataRowConfig.titleParams\">\r\n </div>\r\n <div class=\"text-l\" *ngIf=\"noDataRowConfig.description\"\r\n [innerHtml]=\"noDataRowConfig.description | translate: noDataRowConfig.descriptionParams\"></div>\r\n </ng-container>\r\n <ng-template #loading>\r\n <mat-spinner color=\"accent\"></mat-spinner>\r\n </ng-template>\r\n </div>\r\n </td>\r\n </tr>\r\n </ng-container>\r\n\r\n </table>\r\n</div>\r\n", styles: [":host ::ng-deep th,:host ::ng-deep tr{white-space:nowrap}:host ::ng-deep td{padding-right:10px!important}:host ::ng-deep tr.mat-row:hover,:host ::ng-deep tr.mat-row.highlight{background-color:var(--base-95)}table{width:100%}.column-select-icon{rotate:90deg}.column-select-wrapper{padding:1rem}.material-icons,.material-icons-outlined{margin-right:.5rem}.table-action-button{margin-bottom:1rem}.mat-no-data-row{text-align:center}.mat-no-data-row .no-engines-wrapper{margin-top:4rem}.mat-no-data-row .material-icons-outlined{font-size:4rem;color:var(--base-70)}mat-spinner{margin:auto}th.mat-header-cell:first-of-type,td.mat-cell:first-of-type,td.mat-footer-cell:first-of-type{padding:0 22px}.row-select{width:0}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i4.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i4.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i4.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { kind: "directive", type: i4.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i4.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i4.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i4.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i4.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i4.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i4.MatNoDataRow, selector: "ng-template[matNoDataRow]" }, { kind: "component", type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i6.MatMenu, selector: "mat-menu", exportAs: ["matMenu"] }, { kind: "directive", type: i6.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", exportAs: ["matMenuTrigger"] }, { kind: "component", type: i7.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i8.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i9.FilterBarComponent, selector: "tld-filter-bar", inputs: ["settings", "filterRowVisible"], outputs: ["filterBarChange"] }, { kind: "directive", type: i10.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i10.DefaultLayoutGapDirective, selector: " [fxLayoutGap], [fxLayoutGap.xs], [fxLayoutGap.sm], [fxLayoutGap.md], [fxLayoutGap.lg], [fxLayoutGap.xl], [fxLayoutGap.lt-sm], [fxLayoutGap.lt-md], [fxLayoutGap.lt-lg], [fxLayoutGap.lt-xl], [fxLayoutGap.gt-xs], [fxLayoutGap.gt-sm], [fxLayoutGap.gt-md], [fxLayoutGap.gt-lg]", inputs: ["fxLayoutGap", "fxLayoutGap.xs", "fxLayoutGap.sm", "fxLayoutGap.md", "fxLayoutGap.lg", "fxLayoutGap.xl", "fxLayoutGap.lt-sm", "fxLayoutGap.lt-md", "fxLayoutGap.lt-lg", "fxLayoutGap.lt-xl", "fxLayoutGap.gt-xs", "fxLayoutGap.gt-sm", "fxLayoutGap.gt-md", "fxLayoutGap.gt-lg"] }, { kind: "directive", type: i10.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { kind: "component", type: i11.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
191
|
+
MultiFunctionalTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MultiFunctionalTableComponent, selector: "tld-multi-functional-table", inputs: { config: "config", highlightedElements: "highlightedElements", selection: "selection" }, outputs: { filterBarChange: "filterBarChange", exported: "exported", selectionChange: "selectionChange" }, queries: [{ propertyName: "noDataRow", first: true, predicate: MatNoDataRow, descendants: true }, { propertyName: "headerRowDefs", predicate: MatHeaderRowDef }, { propertyName: "rowDefs", predicate: MatRowDef, descendants: true }, { propertyName: "columnDefs", predicate: MatColumnDef }], viewQueries: [{ propertyName: "table", first: true, predicate: MatTable, descendants: true, static: true }, { propertyName: "sort", first: true, predicate: MatSort, descendants: true }, { propertyName: "tableElementRef", first: true, predicate: MatTable, descendants: true, read: ElementRef }], ngImport: i0, template: "<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n <div fxLayout=\"row\">\r\n <div fxFlex fxLayoutGap=\"1rem\">\r\n <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n <span class=\"material-icons column-select-icon\">menu</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n </button>\r\n\r\n <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n <span class=\"material-icons\">filter_list</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n </button>\r\n\r\n <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n <span class=\"material-icons-outlined\">cloud_download</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n </button>\r\n </div>\r\n\r\n <ng-content select=\"[additionalActions]\"></ng-content>\r\n </div>\r\n\r\n <mat-menu #columnMenu=\"matMenu\">\r\n <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n <div *ngFor=\"let column of configurableColumns\">\r\n <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n {{column.displayName | translate}}\r\n </mat-checkbox>\r\n </div>\r\n </div>\r\n </mat-menu>\r\n\r\n <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n (filterBarChange)=\"filtersChanged($event)\">\r\n </tld-filter-bar>\r\n\r\n <div class=\"table-overflow\">\r\n <table #table mat-table [dataSource]=\"config.dataSource\">\r\n <ng-content></ng-content>\r\n <ng-container [matColumnDef]=\"batchColumnName\">\r\n <th class=\"row-select\" mat-header-cell *matHeaderCellDef disable-export>\r\n <mat-checkbox [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (change)=\"toggleAllRowSelection()\"\r\n [checked]=\"matSelection.hasValue() && isAllSelected()\"\r\n [indeterminate]=\"matSelection.hasValue() && !isAllSelected()\">\r\n </mat-checkbox>\r\n </th>\r\n <td class=\"row-select\" mat-cell *matCellDef=\"let element\" disable-export>\r\n <mat-checkbox *ngIf=\"hoveredRow === element || matSelection.isSelected(element)\" [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (click)=\"$event.stopPropagation()\"\r\n (change)=\"toggleElementSelection(element)\" [checked]=\"matSelection.isSelected(element)\">\r\n </mat-checkbox>\r\n </td>\r\n </ng-container>\r\n <tr mat-header-row *matHeaderRowDef=\"displayColumns\" sticky></tr>\r\n <tr mat-row *matRowDef=\"let row; columns: displayColumns\" (mouseover)=\"hoveredRow = row\" (mouseleave)=\"hoveredRow = null\" [class.highlight]=\"highlightElement(row)\"></tr>\r\n\r\n <ng-container *ngIf=\"noDataRowActive\">\r\n <tr *matNoDataRow>\r\n <!-- add random number to make sure it takes full width -->\r\n <td colspan=\"99\">\r\n <div class=\"no-engines-wrapper\">\r\n <ng-container *ngIf=\"!noDataRowConfig.loading; else loading\">\r\n <div>\r\n <span class=\"material-icons-outlined\">\r\n {{noDataRowIcon}}\r\n </span>\r\n </div>\r\n <div class=\"text-xl-semi-bold\" *ngIf=\"noDataRowConfig.title\"\r\n [innerHtml]=\"noDataRowConfig.title | translate: noDataRowConfig.titleParams\">\r\n </div>\r\n <div class=\"text-l\" *ngIf=\"noDataRowConfig.description\"\r\n [innerHtml]=\"noDataRowConfig.description | translate: noDataRowConfig.descriptionParams\"></div>\r\n </ng-container>\r\n <ng-template #loading>\r\n <mat-spinner color=\"accent\"></mat-spinner>\r\n </ng-template>\r\n </div>\r\n </td>\r\n </tr>\r\n </ng-container>\r\n </table>\r\n </div>\r\n</div>\r\n", styles: [":host ::ng-deep th,:host ::ng-deep tr{white-space:nowrap}:host ::ng-deep td{padding-right:10px!important}:host ::ng-deep tr.mat-row:hover,:host ::ng-deep tr.mat-row.highlight{background-color:var(--base-95)}table{width:100%}.column-select-icon{rotate:90deg}.column-select-wrapper{padding:1rem}.material-icons,.material-icons-outlined{margin-right:.5rem}.table-action-button{margin-bottom:1rem}.mat-no-data-row{text-align:center}.mat-no-data-row .no-engines-wrapper{margin-top:4rem}.mat-no-data-row .material-icons-outlined{font-size:4rem;color:var(--base-70)}mat-spinner{margin:auto}th.mat-header-cell:first-of-type,td.mat-cell:first-of-type,td.mat-footer-cell:first-of-type{padding:0 22px}.row-select{width:0}.mat-checkbox{padding-top:8px}.table-overflow{overflow-x:auto}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i4.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i4.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i4.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { kind: "directive", type: i4.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i4.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i4.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i4.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i4.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i4.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i4.MatNoDataRow, selector: "ng-template[matNoDataRow]" }, { kind: "component", type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i6.MatMenu, selector: "mat-menu", exportAs: ["matMenu"] }, { kind: "directive", type: i6.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", exportAs: ["matMenuTrigger"] }, { kind: "component", type: i7.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i8.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i9.FilterBarComponent, selector: "tld-filter-bar", inputs: ["settings", "filterRowVisible"], outputs: ["filterBarChange"] }, { kind: "directive", type: i10.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i10.DefaultLayoutGapDirective, selector: " [fxLayoutGap], [fxLayoutGap.xs], [fxLayoutGap.sm], [fxLayoutGap.md], [fxLayoutGap.lg], [fxLayoutGap.xl], [fxLayoutGap.lt-sm], [fxLayoutGap.lt-md], [fxLayoutGap.lt-lg], [fxLayoutGap.lt-xl], [fxLayoutGap.gt-xs], [fxLayoutGap.gt-sm], [fxLayoutGap.gt-md], [fxLayoutGap.gt-lg]", inputs: ["fxLayoutGap", "fxLayoutGap.xs", "fxLayoutGap.sm", "fxLayoutGap.md", "fxLayoutGap.lg", "fxLayoutGap.xl", "fxLayoutGap.lt-sm", "fxLayoutGap.lt-md", "fxLayoutGap.lt-lg", "fxLayoutGap.lt-xl", "fxLayoutGap.gt-xs", "fxLayoutGap.gt-sm", "fxLayoutGap.gt-md", "fxLayoutGap.gt-lg"] }, { kind: "directive", type: i10.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { kind: "component", type: i11.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
192
192
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MultiFunctionalTableComponent, decorators: [{
|
|
193
193
|
type: Component,
|
|
194
|
-
args: [{ selector: 'tld-multi-functional-table', template: "<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n <div fxLayout=\"row\">\r\n <div fxFlex fxLayoutGap=\"1rem\">\r\n <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n <span class=\"material-icons column-select-icon\">menu</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n </button>\r\n\r\n <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n <span class=\"material-icons\">filter_list</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n </button>\r\n\r\n <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n <span class=\"material-icons-outlined\">cloud_download</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n </button>\r\n </div>\r\n\r\n <ng-content select=\"[additionalActions]\"></ng-content>\r\n </div>\r\n\r\n <mat-menu #columnMenu=\"matMenu\">\r\n <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n <div *ngFor=\"let column of configurableColumns\">\r\n <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n {{column.displayName | translate}}\r\n </mat-checkbox>\r\n </div>\r\n </div>\r\n </mat-menu>\r\n\r\n <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n (filterBarChange)=\"filtersChanged($event)\">\r\n </tld-filter-bar>\r\n <table #table mat-table [dataSource]=\"config.dataSource\">\r\n
|
|
194
|
+
args: [{ selector: 'tld-multi-functional-table', template: "<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n <div fxLayout=\"row\">\r\n <div fxFlex fxLayoutGap=\"1rem\">\r\n <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n <span class=\"material-icons column-select-icon\">menu</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n </button>\r\n\r\n <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n <span class=\"material-icons\">filter_list</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n </button>\r\n\r\n <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n <span class=\"material-icons-outlined\">cloud_download</span>\r\n <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n </button>\r\n </div>\r\n\r\n <ng-content select=\"[additionalActions]\"></ng-content>\r\n </div>\r\n\r\n <mat-menu #columnMenu=\"matMenu\">\r\n <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n <div *ngFor=\"let column of configurableColumns\">\r\n <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n {{column.displayName | translate}}\r\n </mat-checkbox>\r\n </div>\r\n </div>\r\n </mat-menu>\r\n\r\n <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n (filterBarChange)=\"filtersChanged($event)\">\r\n </tld-filter-bar>\r\n\r\n <div class=\"table-overflow\">\r\n <table #table mat-table [dataSource]=\"config.dataSource\">\r\n <ng-content></ng-content>\r\n <ng-container [matColumnDef]=\"batchColumnName\">\r\n <th class=\"row-select\" mat-header-cell *matHeaderCellDef disable-export>\r\n <mat-checkbox [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (change)=\"toggleAllRowSelection()\"\r\n [checked]=\"matSelection.hasValue() && isAllSelected()\"\r\n [indeterminate]=\"matSelection.hasValue() && !isAllSelected()\">\r\n </mat-checkbox>\r\n </th>\r\n <td class=\"row-select\" mat-cell *matCellDef=\"let element\" disable-export>\r\n <mat-checkbox *ngIf=\"hoveredRow === element || matSelection.isSelected(element)\" [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (click)=\"$event.stopPropagation()\"\r\n (change)=\"toggleElementSelection(element)\" [checked]=\"matSelection.isSelected(element)\">\r\n </mat-checkbox>\r\n </td>\r\n </ng-container>\r\n <tr mat-header-row *matHeaderRowDef=\"displayColumns\" sticky></tr>\r\n <tr mat-row *matRowDef=\"let row; columns: displayColumns\" (mouseover)=\"hoveredRow = row\" (mouseleave)=\"hoveredRow = null\" [class.highlight]=\"highlightElement(row)\"></tr>\r\n\r\n <ng-container *ngIf=\"noDataRowActive\">\r\n <tr *matNoDataRow>\r\n <!-- add random number to make sure it takes full width -->\r\n <td colspan=\"99\">\r\n <div class=\"no-engines-wrapper\">\r\n <ng-container *ngIf=\"!noDataRowConfig.loading; else loading\">\r\n <div>\r\n <span class=\"material-icons-outlined\">\r\n {{noDataRowIcon}}\r\n </span>\r\n </div>\r\n <div class=\"text-xl-semi-bold\" *ngIf=\"noDataRowConfig.title\"\r\n [innerHtml]=\"noDataRowConfig.title | translate: noDataRowConfig.titleParams\">\r\n </div>\r\n <div class=\"text-l\" *ngIf=\"noDataRowConfig.description\"\r\n [innerHtml]=\"noDataRowConfig.description | translate: noDataRowConfig.descriptionParams\"></div>\r\n </ng-container>\r\n <ng-template #loading>\r\n <mat-spinner color=\"accent\"></mat-spinner>\r\n </ng-template>\r\n </div>\r\n </td>\r\n </tr>\r\n </ng-container>\r\n </table>\r\n </div>\r\n</div>\r\n", styles: [":host ::ng-deep th,:host ::ng-deep tr{white-space:nowrap}:host ::ng-deep td{padding-right:10px!important}:host ::ng-deep tr.mat-row:hover,:host ::ng-deep tr.mat-row.highlight{background-color:var(--base-95)}table{width:100%}.column-select-icon{rotate:90deg}.column-select-wrapper{padding:1rem}.material-icons,.material-icons-outlined{margin-right:.5rem}.table-action-button{margin-bottom:1rem}.mat-no-data-row{text-align:center}.mat-no-data-row .no-engines-wrapper{margin-top:4rem}.mat-no-data-row .material-icons-outlined{font-size:4rem;color:var(--base-70)}mat-spinner{margin:auto}th.mat-header-cell:first-of-type,td.mat-cell:first-of-type,td.mat-footer-cell:first-of-type{padding:0 22px}.row-select{width:0}.mat-checkbox{padding-top:8px}.table-overflow{overflow-x:auto}\n"] }]
|
|
195
195
|
}], ctorParameters: function () { return [{ type: i1.DOMService }, { type: i2.TranslateService }]; }, propDecorators: { config: [{
|
|
196
196
|
type: Input
|
|
197
197
|
}], highlightedElements: [{
|
|
@@ -226,4 +226,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
226
226
|
type: ViewChild,
|
|
227
227
|
args: [MatTable, { read: ElementRef }]
|
|
228
228
|
}] } });
|
|
229
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-functional-table.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-common/src/lib/multi-functional-table/multi-functional-table.component.ts","../../../../../projects/ngx-common/src/lib/multi-functional-table/multi-functional-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { SelectionModel } from '@angular/cdk/collections';\r\nimport { AfterContentInit, AfterViewInit, Component, ContentChild, ContentChildren, ElementRef, EventEmitter, Input, OnInit, Output, QueryList, ViewChild } from '@angular/core';\r\nimport { MatSort } from '@angular/material/sort';\r\nimport { MatColumnDef, MatHeaderRowDef, MatNoDataRow, MatRowDef, MatTable } from '@angular/material/table';\r\nimport { TranslateService } from '@ngx-translate/core';\r\nimport { map, Observable, take } from 'rxjs';\r\nimport { FilterBarSettings } from '../filter-bar';\r\nimport { FilterBarChangeEvent } from '../filter-bar/models/filter-bar-change-event.model';\r\nimport { HtmlElementParseHelper } from '../helpers';\r\nimport { SaveFileHelper } from '../helpers/save-file.helper';\r\nimport { DOMService } from '../services';\r\nimport { DISABLE_EXPORT_ATTRIBUTE_NAME } from './constants/disable-export-attribute-name.const';\r\nimport { ExportFormat, NoDataRowConfig } from './models';\r\nimport { ColumnConfig } from './models/column-config.model';\r\nimport { MultiFunctionalTableConfig } from './models/multi-functional-table-config.model';\r\nimport { SelectedColumnLocalStorageConfig } from './models/selected-column-local-storage-config.model';\r\n\r\n@Component({\r\n  // eslint-disable-next-line @angular-eslint/component-selector\r\n  selector: 'tld-multi-functional-table',\r\n  templateUrl: './multi-functional-table.component.html',\r\n  styleUrls: ['./multi-functional-table.component.scss']\r\n})\r\nexport class MultiFunctionalTableComponent<T> implements OnInit, AfterContentInit, AfterViewInit {\r\n  readonly matSelection = new SelectionModel<T>(true, []);\r\n\r\n  @Input() config!: MultiFunctionalTableConfig<T>;\r\n  @Input() highlightedElements!: T[];\r\n  @Input() set selection(value: T[]) {\r\n    this.matSelection.setSelection(...(value ?? []));\r\n  }\r\n\r\n  //#region Output properties\r\n  @Output() filterBarChange: EventEmitter<FilterBarChangeEvent> = new EventEmitter<FilterBarChangeEvent>();\r\n  @Output() exported: EventEmitter<T[]> = new EventEmitter<T[]>();\r\n  @Output() selectionChange: Observable<T[]> = this.matSelection.changed.asObservable().pipe(map(() => { return this.matSelection.selected; }));\r\n  //#endregion\r\n\r\n  // #region Properties for mat table wrapper\r\n  @ContentChildren(MatHeaderRowDef) headerRowDefs!: QueryList<MatHeaderRowDef>;\r\n  @ContentChildren(MatRowDef, { descendants: true }) rowDefs!: QueryList<MatRowDef<T>>;\r\n  @ContentChildren(MatColumnDef) columnDefs!: QueryList<MatColumnDef>;\r\n  @ContentChild(MatNoDataRow) noDataRow!: MatNoDataRow;\r\n\r\n  @ViewChild(MatTable, { static: true }) table!: MatTable<T>;\r\n  @ViewChild(MatSort) sort!: MatSort;\r\n  //#endregion\r\n\r\n  @ViewChild(MatTable, { read: ElementRef }) tableElementRef!: ElementRef<HTMLElement>;\r\n\r\n\r\n  get ignoreAttributeName() { return DISABLE_EXPORT_ATTRIBUTE_NAME }\r\n  displayColumns!: string[];\r\n\r\n  columnSelectActive!: boolean;\r\n  get filterActive() { return this.config.filter?.enabled; }\r\n  exportActive!: boolean;\r\n  noDataRowActive!: boolean;\r\n\r\n  filterEnabled!: boolean;\r\n  filterBarVisible!: boolean;\r\n  filterSettings!: FilterBarSettings;\r\n\r\n  noDataRowConfig!: NoDataRowConfig;\r\n  noDataRowIcon = \"manage_search\";\r\n\r\n  configurableColumns!: ColumnConfig[];\r\n  batchSelectedEnabled!: boolean;\r\n\r\n  // is used in html to display batch checkbox on hovered row\r\n  hoveredRow: any;\r\n\r\n  private localStorageValue!: SelectedColumnLocalStorageConfig;\r\n  readonly batchColumnName = \"batch\";\r\n\r\n  constructor(private readonly domService: DOMService,\r\n    private readonly translateService: TranslateService) { }\r\n\r\n  //#region Angular lifecycle hooks\r\n  ngOnInit(): void {\r\n    this.readFromLocalStorage();\r\n    this.setFilterProperties();\r\n    this.setColumnSelectProperties();\r\n    this.setExportProperties();\r\n    this.setNoDataRowProperties();\r\n  }\r\n\r\n  ngAfterContentInit() {\r\n\r\n    this.columnDefs.forEach(columnDef => this.table.addColumnDef(columnDef));\r\n    this.rowDefs.forEach(rowDef => this.table.addRowDef(rowDef));\r\n    this.headerRowDefs.forEach(headerRowDef => this.table.addHeaderRowDef(headerRowDef));\r\n    this.headerRowDefs.forEach(headerRowDef => this.table.addHeaderRowDef(headerRowDef));\r\n    this.table.setNoDataRow(this.noDataRow);\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.config.dataSource.sort = this.sort;\r\n  }\r\n  //#endregion\r\n\r\n  updateDisplayColumns(initial = false) {\r\n    this.batchSelectedEnabled = this.config.batchConfig?.enabled ? true : false;\r\n    const allColumns = this.config.columnSelect?.columns ?? [];\r\n    this.displayColumns = [];\r\n    this.configurableColumns = allColumns.filter(column => !column.notConfigurable);\r\n\r\n    if (this.batchSelectedEnabled) {\r\n      this.displayColumns.push(this.batchColumnName);\r\n    }\r\n\r\n    if (!allColumns.length) {\r\n      return;\r\n    }\r\n\r\n    const localStorageValue = this.localStorageValue ?? {};\r\n    if (!localStorageValue.columns) {\r\n      localStorageValue.columns = {};\r\n    }\r\n\r\n    allColumns.forEach((column) => {\r\n      if (initial) {\r\n        column.selected = localStorageValue.columns?.[column.name] ?? column.selected;\r\n      }\r\n      else {\r\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n        localStorageValue.columns![column.name] = column.selected;\r\n      }\r\n      if (column.selected) {\r\n        this.displayColumns.push(column.name);\r\n      }\r\n    })\r\n    if (!initial) {\r\n      this.updateLocalStorage(localStorageValue);\r\n    }\r\n\r\n\r\n  }\r\n\r\n  toggleFilterBar() {\r\n    const localstorageObject = this.localStorageValue ?? {};\r\n    if (!localstorageObject.filter) {\r\n      localstorageObject.filter = {};\r\n    }\r\n\r\n    this.filterBarVisible = !this.filterBarVisible;\r\n    localstorageObject.filter.visible = this.filterBarVisible;\r\n    this.updateLocalStorage(localstorageObject);\r\n  }\r\n\r\n  filtersChanged(filters: FilterBarChangeEvent) {\r\n    this.filterBarChange.emit(filters);\r\n  }\r\n\r\n  export() {\r\n    this.config.dataSource.connect().pipe(take(1)).subscribe((data) => {\r\n      this.exported.emit(data);\r\n    })\r\n\r\n    if (this.config.export?.fileOptions) {\r\n      this.exportToFile()\r\n    }\r\n  }\r\n\r\n  highlightElement(element: T) {\r\n    return this.highlightedElements?.includes(element);\r\n  }\r\n\r\n  //#region Methods for batch selection\r\n  isAllSelected() {\r\n    const numSelected = this.matSelection.selected.length;\r\n    const numRows = this.config.dataSource.data.length;\r\n    return numSelected === numRows;\r\n  }\r\n\r\n  toggleAllRowSelection() {\r\n    if (this.isAllSelected()) {\r\n      this.matSelection.clear();\r\n      return;\r\n    }\r\n\r\n    this.matSelection.select(...this.config.dataSource.data);\r\n  }\r\n\r\n  toggleElementSelection(element: T) {\r\n    this.matSelection.toggle(element);\r\n  }\r\n  //#endregion\r\n\r\n  private exportToFile() {\r\n    if (!this.config.export?.fileOptions?.saveToFile) {\r\n      return;\r\n    }\r\n\r\n    let fileString = \"\";\r\n    let extension!: ExportFormat;\r\n    switch (this.config.export.fileOptions.format) {\r\n      case ExportFormat.CSV:\r\n      default:\r\n        extension = ExportFormat.CSV;\r\n        fileString = HtmlElementParseHelper.tableAsCsv(this.tableElementRef.nativeElement, this.ignoreAttributeName);\r\n        break;\r\n    }\r\n    const translatedFileName = this.translateService.instant(this.config.export.fileOptions.fileName);\r\n    SaveFileHelper.saveFile(fileString, `${translatedFileName}.${extension}`);\r\n  }\r\n\r\n  private updateLocalStorage(newValue: SelectedColumnLocalStorageConfig) {\r\n    this.localStorageValue = newValue;\r\n    if (this.config.localStorageKey && newValue) {\r\n      this.domService.localStorage?.setItem(this.config.localStorageKey, JSON.stringify(newValue));\r\n    }\r\n  }\r\n  private readFromLocalStorage() {\r\n    if (this.config.localStorageKey) {\r\n      try {\r\n        const localStorageString = this.domService.localStorage?.getItem(this.config.localStorageKey);\r\n        if (localStorageString) {\r\n          this.localStorageValue = JSON.parse(localStorageString);\r\n        }\r\n      }\r\n      // eslint-disable-next-line no-empty\r\n      catch {\r\n      }\r\n    }\r\n  }\r\n\r\n  private setColumnSelectProperties() {\r\n    this.columnSelectActive = this.config.columnSelect?.enabled ? true : false;\r\n    this.updateDisplayColumns(true);\r\n  }\r\n\r\n  private setFilterProperties() {\r\n    if (this.config.filter?.enabled) {\r\n      this.filterEnabled = true;\r\n      this.filterBarVisible = this.localStorageValue?.filter?.visible ?? this.config.filter.visible ?? false;\r\n      this.filterSettings = this.config.filter.settings;\r\n    }\r\n  }\r\n\r\n  private setNoDataRowProperties() {\r\n    if (this.config.noDataRow) {\r\n      this.noDataRowActive = true;\r\n      this.noDataRowConfig = this.config.noDataRow;\r\n      if (this.noDataRowConfig.icon) {\r\n        this.noDataRowIcon = this.noDataRowConfig.icon;\r\n      }\r\n    }\r\n  }\r\n\r\n  private setExportProperties() {\r\n    if (this.config.export?.enabled) {\r\n      this.exportActive = true;\r\n    }\r\n  }\r\n}\r\n","<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n  <div fxLayout=\"row\">\r\n    <div fxFlex fxLayoutGap=\"1rem\">\r\n      <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n        <span class=\"material-icons column-select-icon\">menu</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n      </button>\r\n\r\n      <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n        <span class=\"material-icons\">filter_list</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n      </button>\r\n\r\n      <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n        <span class=\"material-icons-outlined\">cloud_download</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n      </button>\r\n    </div>\r\n\r\n    <ng-content select=\"[additionalActions]\"></ng-content>\r\n  </div>\r\n\r\n  <mat-menu #columnMenu=\"matMenu\">\r\n    <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n      <div *ngFor=\"let column of configurableColumns\">\r\n        <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n          {{column.displayName | translate}}\r\n        </mat-checkbox>\r\n      </div>\r\n    </div>\r\n  </mat-menu>\r\n\r\n  <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n    (filterBarChange)=\"filtersChanged($event)\">\r\n  </tld-filter-bar>\r\n  <table #table mat-table [dataSource]=\"config.dataSource\">\r\n    <ng-content></ng-content>\r\n    <ng-container [matColumnDef]=\"batchColumnName\">\r\n      <th class=\"row-select\" mat-header-cell *matHeaderCellDef disable-export>\r\n        <mat-checkbox [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (change)=\"toggleAllRowSelection()\"\r\n          [checked]=\"matSelection.hasValue() && isAllSelected()\"\r\n          [indeterminate]=\"matSelection.hasValue() && !isAllSelected()\">\r\n        </mat-checkbox>\r\n      </th>\r\n      <td class=\"row-select\" mat-cell *matCellDef=\"let element\" disable-export>\r\n        <mat-checkbox *ngIf=\"hoveredRow === element || matSelection.isSelected(element)\" [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (click)=\"$event.stopPropagation()\"\r\n          (change)=\"toggleElementSelection(element)\" [checked]=\"matSelection.isSelected(element)\">\r\n        </mat-checkbox>\r\n      </td>\r\n    </ng-container>\r\n    <tr mat-header-row *matHeaderRowDef=\"displayColumns\" sticky></tr>\r\n    <tr mat-row *matRowDef=\"let row; columns: displayColumns\" (mouseover)=\"hoveredRow = row\" (mouseleave)=\"hoveredRow = null\" [class.highlight]=\"highlightElement(row)\"></tr>\r\n\r\n    <ng-container *ngIf=\"noDataRowActive\">\r\n      <tr *matNoDataRow>\r\n        <!-- add random number to make sure it takes full width -->\r\n        <td colspan=\"99\">\r\n          <div class=\"no-engines-wrapper\">\r\n            <ng-container *ngIf=\"!noDataRowConfig.loading; else loading\">\r\n              <div>\r\n                <span class=\"material-icons-outlined\">\r\n                  {{noDataRowIcon}}\r\n                </span>\r\n              </div>\r\n              <div class=\"text-xl-semi-bold\" *ngIf=\"noDataRowConfig.title\"\r\n                [innerHtml]=\"noDataRowConfig.title | translate: noDataRowConfig.titleParams\">\r\n              </div>\r\n              <div class=\"text-l\" *ngIf=\"noDataRowConfig.description\"\r\n                [innerHtml]=\"noDataRowConfig.description | translate: noDataRowConfig.descriptionParams\"></div>\r\n            </ng-container>\r\n            <ng-template #loading>\r\n              <mat-spinner color=\"accent\"></mat-spinner>\r\n            </ng-template>\r\n          </div>\r\n        </td>\r\n      </tr>\r\n    </ng-container>\r\n\r\n  </table>\r\n</div>\r\n"]}
|
|
229
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-functional-table.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-common/src/lib/multi-functional-table/multi-functional-table.component.ts","../../../../../projects/ngx-common/src/lib/multi-functional-table/multi-functional-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { SelectionModel } from '@angular/cdk/collections';\r\nimport { AfterContentInit, AfterViewInit, Component, ContentChild, ContentChildren, ElementRef, EventEmitter, Input, OnInit, Output, QueryList, ViewChild } from '@angular/core';\r\nimport { MatSort } from '@angular/material/sort';\r\nimport { MatColumnDef, MatHeaderRowDef, MatNoDataRow, MatRowDef, MatTable } from '@angular/material/table';\r\nimport { TranslateService } from '@ngx-translate/core';\r\nimport { map, Observable, take } from 'rxjs';\r\nimport { FilterBarSettings } from '../filter-bar';\r\nimport { FilterBarChangeEvent } from '../filter-bar/models/filter-bar-change-event.model';\r\nimport { HtmlElementParseHelper } from '../helpers';\r\nimport { SaveFileHelper } from '../helpers/save-file.helper';\r\nimport { DOMService } from '../services';\r\nimport { DISABLE_EXPORT_ATTRIBUTE_NAME } from './constants/disable-export-attribute-name.const';\r\nimport { ExportFormat, NoDataRowConfig } from './models';\r\nimport { ColumnConfig } from './models/column-config.model';\r\nimport { MultiFunctionalTableConfig } from './models/multi-functional-table-config.model';\r\nimport { SelectedColumnLocalStorageConfig } from './models/selected-column-local-storage-config.model';\r\n\r\n@Component({\r\n  // eslint-disable-next-line @angular-eslint/component-selector\r\n  selector: 'tld-multi-functional-table',\r\n  templateUrl: './multi-functional-table.component.html',\r\n  styleUrls: ['./multi-functional-table.component.scss']\r\n})\r\nexport class MultiFunctionalTableComponent<T> implements OnInit, AfterContentInit, AfterViewInit {\r\n  readonly matSelection = new SelectionModel<T>(true, []);\r\n\r\n  @Input() config!: MultiFunctionalTableConfig<T>;\r\n  @Input() highlightedElements!: T[];\r\n  @Input() set selection(value: T[]) {\r\n    this.matSelection.setSelection(...(value ?? []));\r\n  }\r\n\r\n  //#region Output properties\r\n  @Output() filterBarChange: EventEmitter<FilterBarChangeEvent> = new EventEmitter<FilterBarChangeEvent>();\r\n  @Output() exported: EventEmitter<T[]> = new EventEmitter<T[]>();\r\n  @Output() selectionChange: Observable<T[]> = this.matSelection.changed.asObservable().pipe(map(() => { return this.matSelection.selected; }));\r\n  //#endregion\r\n\r\n  // #region Properties for mat table wrapper\r\n  @ContentChildren(MatHeaderRowDef) headerRowDefs!: QueryList<MatHeaderRowDef>;\r\n  @ContentChildren(MatRowDef, { descendants: true }) rowDefs!: QueryList<MatRowDef<T>>;\r\n  @ContentChildren(MatColumnDef) columnDefs!: QueryList<MatColumnDef>;\r\n  @ContentChild(MatNoDataRow) noDataRow!: MatNoDataRow;\r\n\r\n  @ViewChild(MatTable, { static: true }) table!: MatTable<T>;\r\n  @ViewChild(MatSort) sort!: MatSort;\r\n  //#endregion\r\n\r\n  @ViewChild(MatTable, { read: ElementRef }) tableElementRef!: ElementRef<HTMLElement>;\r\n\r\n\r\n  get ignoreAttributeName() { return DISABLE_EXPORT_ATTRIBUTE_NAME }\r\n  displayColumns!: string[];\r\n\r\n  columnSelectActive!: boolean;\r\n  get filterActive() { return this.config.filter?.enabled; }\r\n  exportActive!: boolean;\r\n  noDataRowActive!: boolean;\r\n\r\n  filterEnabled!: boolean;\r\n  filterBarVisible!: boolean;\r\n  filterSettings!: FilterBarSettings;\r\n\r\n  noDataRowConfig!: NoDataRowConfig;\r\n  noDataRowIcon = \"manage_search\";\r\n\r\n  configurableColumns!: ColumnConfig[];\r\n  batchSelectedEnabled!: boolean;\r\n\r\n  // is used in html to display batch checkbox on hovered row\r\n  hoveredRow: any;\r\n\r\n  private localStorageValue!: SelectedColumnLocalStorageConfig;\r\n  readonly batchColumnName = \"batch\";\r\n\r\n  constructor(private readonly domService: DOMService,\r\n    private readonly translateService: TranslateService) { }\r\n\r\n  //#region Angular lifecycle hooks\r\n  ngOnInit(): void {\r\n    this.readFromLocalStorage();\r\n    this.setFilterProperties();\r\n    this.setColumnSelectProperties();\r\n    this.setExportProperties();\r\n    this.setNoDataRowProperties();\r\n  }\r\n\r\n  ngAfterContentInit() {\r\n\r\n    this.columnDefs.forEach(columnDef => this.table.addColumnDef(columnDef));\r\n    this.rowDefs.forEach(rowDef => this.table.addRowDef(rowDef));\r\n    this.headerRowDefs.forEach(headerRowDef => this.table.addHeaderRowDef(headerRowDef));\r\n    this.headerRowDefs.forEach(headerRowDef => this.table.addHeaderRowDef(headerRowDef));\r\n    this.table.setNoDataRow(this.noDataRow);\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.config.dataSource.sort = this.sort;\r\n  }\r\n  //#endregion\r\n\r\n  updateDisplayColumns(initial = false) {\r\n    this.batchSelectedEnabled = this.config.batchConfig?.enabled ? true : false;\r\n    const allColumns = this.config.columnSelect?.columns ?? [];\r\n    this.displayColumns = [];\r\n    this.configurableColumns = allColumns.filter(column => !column.notConfigurable);\r\n\r\n    if (this.batchSelectedEnabled) {\r\n      this.displayColumns.push(this.batchColumnName);\r\n    }\r\n\r\n    if (!allColumns.length) {\r\n      return;\r\n    }\r\n\r\n    const localStorageValue = this.localStorageValue ?? {};\r\n    if (!localStorageValue.columns) {\r\n      localStorageValue.columns = {};\r\n    }\r\n\r\n    allColumns.forEach((column) => {\r\n      if (initial) {\r\n        column.selected = localStorageValue.columns?.[column.name] ?? column.selected;\r\n      }\r\n      else {\r\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n        localStorageValue.columns![column.name] = column.selected;\r\n      }\r\n      if (column.selected) {\r\n        this.displayColumns.push(column.name);\r\n      }\r\n    })\r\n    if (!initial) {\r\n      this.updateLocalStorage(localStorageValue);\r\n    }\r\n\r\n\r\n  }\r\n\r\n  toggleFilterBar() {\r\n    const localstorageObject = this.localStorageValue ?? {};\r\n    if (!localstorageObject.filter) {\r\n      localstorageObject.filter = {};\r\n    }\r\n\r\n    this.filterBarVisible = !this.filterBarVisible;\r\n    localstorageObject.filter.visible = this.filterBarVisible;\r\n    this.updateLocalStorage(localstorageObject);\r\n  }\r\n\r\n  filtersChanged(filters: FilterBarChangeEvent) {\r\n    this.filterBarChange.emit(filters);\r\n  }\r\n\r\n  export() {\r\n    this.config.dataSource.connect().pipe(take(1)).subscribe((data) => {\r\n      this.exported.emit(data);\r\n    })\r\n\r\n    if (this.config.export?.fileOptions) {\r\n      this.exportToFile()\r\n    }\r\n  }\r\n\r\n  highlightElement(element: T) {\r\n    return this.highlightedElements?.includes(element);\r\n  }\r\n\r\n  //#region Methods for batch selection\r\n  isAllSelected() {\r\n    const numSelected = this.matSelection.selected.length;\r\n    const numRows = this.config.dataSource.data.length;\r\n    return numSelected === numRows;\r\n  }\r\n\r\n  toggleAllRowSelection() {\r\n    if (this.isAllSelected()) {\r\n      this.matSelection.clear();\r\n      return;\r\n    }\r\n\r\n    this.matSelection.select(...this.config.dataSource.data);\r\n  }\r\n\r\n  toggleElementSelection(element: T) {\r\n    this.matSelection.toggle(element);\r\n  }\r\n  //#endregion\r\n\r\n  private exportToFile() {\r\n    if (!this.config.export?.fileOptions?.saveToFile) {\r\n      return;\r\n    }\r\n\r\n    let fileString = \"\";\r\n    let extension!: ExportFormat;\r\n    switch (this.config.export.fileOptions.format) {\r\n      case ExportFormat.CSV:\r\n      default:\r\n        extension = ExportFormat.CSV;\r\n        fileString = HtmlElementParseHelper.tableAsCsv(this.tableElementRef.nativeElement, this.ignoreAttributeName);\r\n        break;\r\n    }\r\n    const translatedFileName = this.translateService.instant(this.config.export.fileOptions.fileName);\r\n    SaveFileHelper.saveFile(fileString, `${translatedFileName}.${extension}`);\r\n  }\r\n\r\n  private updateLocalStorage(newValue: SelectedColumnLocalStorageConfig) {\r\n    this.localStorageValue = newValue;\r\n    if (this.config.localStorageKey && newValue) {\r\n      this.domService.localStorage?.setItem(this.config.localStorageKey, JSON.stringify(newValue));\r\n    }\r\n  }\r\n  private readFromLocalStorage() {\r\n    if (this.config.localStorageKey) {\r\n      try {\r\n        const localStorageString = this.domService.localStorage?.getItem(this.config.localStorageKey);\r\n        if (localStorageString) {\r\n          this.localStorageValue = JSON.parse(localStorageString);\r\n        }\r\n      }\r\n      // eslint-disable-next-line no-empty\r\n      catch {\r\n      }\r\n    }\r\n  }\r\n\r\n  private setColumnSelectProperties() {\r\n    this.columnSelectActive = this.config.columnSelect?.enabled ? true : false;\r\n    this.updateDisplayColumns(true);\r\n  }\r\n\r\n  private setFilterProperties() {\r\n    if (this.config.filter?.enabled) {\r\n      this.filterEnabled = true;\r\n      this.filterBarVisible = this.localStorageValue?.filter?.visible ?? this.config.filter.visible ?? false;\r\n      this.filterSettings = this.config.filter.settings;\r\n    }\r\n  }\r\n\r\n  private setNoDataRowProperties() {\r\n    if (this.config.noDataRow) {\r\n      this.noDataRowActive = true;\r\n      this.noDataRowConfig = this.config.noDataRow;\r\n      if (this.noDataRowConfig.icon) {\r\n        this.noDataRowIcon = this.noDataRowConfig.icon;\r\n      }\r\n    }\r\n  }\r\n\r\n  private setExportProperties() {\r\n    if (this.config.export?.enabled) {\r\n      this.exportActive = true;\r\n    }\r\n  }\r\n}\r\n","<div fxLayout=\"column\" fxLayoutGap=\"1rem\">\r\n\r\n  <div fxLayout=\"row\">\r\n    <div fxFlex fxLayoutGap=\"1rem\">\r\n      <button mat-button [matMenuTriggerFor]=\"columnMenu\" *ngIf=\"columnSelectActive\">\r\n        <span class=\"material-icons column-select-icon\">menu</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.COLUMN_SELECT' | translate}}</span>\r\n      </button>\r\n\r\n      <button mat-button *ngIf=\"filterActive\" (click)=\"toggleFilterBar()\">\r\n        <span class=\"material-icons\">filter_list</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.FILTER_TOGGLE' | translate}}</span>\r\n      </button>\r\n\r\n      <button mat-stroked-button *ngIf=\"exportActive\" (click)=\"export()\">\r\n        <span class=\"material-icons-outlined\">cloud_download</span>\r\n        <span>{{'MULTI_FUNCTIONAL_TABLE.EXPORT' | translate}}</span>\r\n      </button>\r\n    </div>\r\n\r\n    <ng-content select=\"[additionalActions]\"></ng-content>\r\n  </div>\r\n\r\n  <mat-menu #columnMenu=\"matMenu\">\r\n    <div class=\"column-select-wrapper\" (click)=\"$event.stopPropagation()\">\r\n      <div *ngFor=\"let column of configurableColumns\">\r\n        <mat-checkbox [(ngModel)]=\"column.selected\" (change)=\"updateDisplayColumns()\">\r\n          {{column.displayName | translate}}\r\n        </mat-checkbox>\r\n      </div>\r\n    </div>\r\n  </mat-menu>\r\n\r\n  <tld-filter-bar [filterRowVisible]=\"!filterBarVisible\" [settings]=\"filterSettings\" *ngIf=\"filterEnabled\"\r\n    (filterBarChange)=\"filtersChanged($event)\">\r\n  </tld-filter-bar>\r\n\r\n  <div class=\"table-overflow\">\r\n    <table #table mat-table [dataSource]=\"config.dataSource\">\r\n      <ng-content></ng-content>\r\n      <ng-container [matColumnDef]=\"batchColumnName\">\r\n        <th class=\"row-select\" mat-header-cell *matHeaderCellDef disable-export>\r\n          <mat-checkbox [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (change)=\"toggleAllRowSelection()\"\r\n            [checked]=\"matSelection.hasValue() && isAllSelected()\"\r\n            [indeterminate]=\"matSelection.hasValue() && !isAllSelected()\">\r\n          </mat-checkbox>\r\n        </th>\r\n        <td class=\"row-select\" mat-cell *matCellDef=\"let element\" disable-export>\r\n          <mat-checkbox *ngIf=\"hoveredRow === element || matSelection.isSelected(element)\" [disabled]=\"config.batchConfig!.checkBoxesDisabled\" (click)=\"$event.stopPropagation()\"\r\n            (change)=\"toggleElementSelection(element)\" [checked]=\"matSelection.isSelected(element)\">\r\n          </mat-checkbox>\r\n        </td>\r\n      </ng-container>\r\n      <tr mat-header-row *matHeaderRowDef=\"displayColumns\" sticky></tr>\r\n      <tr mat-row *matRowDef=\"let row; columns: displayColumns\" (mouseover)=\"hoveredRow = row\" (mouseleave)=\"hoveredRow = null\" [class.highlight]=\"highlightElement(row)\"></tr>\r\n\r\n      <ng-container *ngIf=\"noDataRowActive\">\r\n        <tr *matNoDataRow>\r\n          <!-- add random number to make sure it takes full width -->\r\n          <td colspan=\"99\">\r\n            <div class=\"no-engines-wrapper\">\r\n              <ng-container *ngIf=\"!noDataRowConfig.loading; else loading\">\r\n                <div>\r\n                  <span class=\"material-icons-outlined\">\r\n                    {{noDataRowIcon}}\r\n                  </span>\r\n                </div>\r\n                <div class=\"text-xl-semi-bold\" *ngIf=\"noDataRowConfig.title\"\r\n                  [innerHtml]=\"noDataRowConfig.title | translate: noDataRowConfig.titleParams\">\r\n                </div>\r\n                <div class=\"text-l\" *ngIf=\"noDataRowConfig.description\"\r\n                  [innerHtml]=\"noDataRowConfig.description | translate: noDataRowConfig.descriptionParams\"></div>\r\n              </ng-container>\r\n              <ng-template #loading>\r\n                <mat-spinner color=\"accent\"></mat-spinner>\r\n              </ng-template>\r\n            </div>\r\n          </td>\r\n        </tr>\r\n      </ng-container>\r\n    </table>\r\n  </div>\r\n</div>\r\n"]}
|