@mediusinc/mng-commons 5.4.0-rc.3 → 5.4.0-rc.5

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.
Files changed (60) hide show
  1. package/core/data-list/data-list-params-helpers.d.ts +1 -1
  2. package/core/data-list/data-list.model.d.ts +1 -0
  3. package/core/helpers/type-helpers.d.ts +12 -0
  4. package/esm2022/core/data-list/data-list-params-helpers.mjs +29 -5
  5. package/esm2022/core/data-list/data-list.model.mjs +1 -1
  6. package/esm2022/core/helpers/type-helpers.mjs +1 -1
  7. package/esm2022/filter/descriptors/filter-generic.descriptor.mjs +21 -0
  8. package/esm2022/filter/descriptors/filter-lookup.descriptor.mjs +1 -1
  9. package/esm2022/filter/descriptors/filter.descriptor.mjs +40 -1
  10. package/esm2022/filter/index.mjs +3 -1
  11. package/esm2022/filter/models/filter-generic-property.model.mjs +2 -0
  12. package/esm2022/form/components/dropdown/dropdown.component.mjs +159 -10
  13. package/esm2022/table/api/descriptors/table.descriptor.mjs +51 -2
  14. package/esm2022/table/components/column-filter/column-filter.component.mjs +2 -2
  15. package/esm2022/table/components/filter/filter-active-tag/filter-active-tag.component.mjs +4 -1
  16. package/esm2022/table/components/filter/filter-form/filter-form.component.mjs +42 -9
  17. package/esm2022/table/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.mjs +95 -11
  18. package/esm2022/table/components/table/table.component.mjs +57 -13
  19. package/esm2022/table/helpers/filters.mjs +32 -2
  20. package/esm2022/table/services/table-feature-config.token.mjs +1 -1
  21. package/esm2022/tableview/action/components/localization/data-language-dropdown.component.mjs +2 -2
  22. package/esm2022/tableview/api/editor/descriptors/field-many.descriptor.mjs +11 -10
  23. package/esm2022/tableview/api/editor/descriptors/field.descriptor.mjs +16 -2
  24. package/esm2022/tableview/api/editor/models/formly-custom-field.model.mjs +2 -1
  25. package/esm2022/tableview/editor/components/formly/fields/formly-field-datepicker/formly-field-datepicker.component.mjs +23 -3
  26. package/esm2022/tableview/editor/components/formly/fields/formly-field-dropdown/formly-field-dropdown.component.mjs +2 -2
  27. package/esm2022/tableview/editor/components/formly/fields/formly-field-input/formly-field-input.component.mjs +3 -3
  28. package/esm2022/tableview/editor/components/formly/fields/formly-field-table-dialog-multiselect/formly-field-table-dialog-multiselect.component.mjs +3 -4
  29. package/esm2022/tableview/editor/components/formly/fields/formly-field-type.abstract.component.mjs +2 -1
  30. package/fesm2022/mediusinc-mng-commons-core.mjs +28 -4
  31. package/fesm2022/mediusinc-mng-commons-core.mjs.map +1 -1
  32. package/fesm2022/mediusinc-mng-commons-filter.mjs +61 -1
  33. package/fesm2022/mediusinc-mng-commons-filter.mjs.map +1 -1
  34. package/fesm2022/mediusinc-mng-commons-form.mjs +156 -7
  35. package/fesm2022/mediusinc-mng-commons-form.mjs.map +1 -1
  36. package/fesm2022/mediusinc-mng-commons-table-api.mjs +50 -1
  37. package/fesm2022/mediusinc-mng-commons-table-api.mjs.map +1 -1
  38. package/fesm2022/mediusinc-mng-commons-table.mjs +230 -45
  39. package/fesm2022/mediusinc-mng-commons-table.mjs.map +1 -1
  40. package/fesm2022/mediusinc-mng-commons-tableview-api.mjs +26 -10
  41. package/fesm2022/mediusinc-mng-commons-tableview-api.mjs.map +1 -1
  42. package/fesm2022/mediusinc-mng-commons-tableview.mjs +29 -8
  43. package/fesm2022/mediusinc-mng-commons-tableview.mjs.map +1 -1
  44. package/filter/descriptors/filter-generic.descriptor.d.ts +11 -0
  45. package/filter/descriptors/filter.descriptor.d.ts +21 -2
  46. package/filter/index.d.ts +2 -0
  47. package/filter/models/filter-generic-property.model.d.ts +7 -0
  48. package/form/components/dropdown/dropdown.component.d.ts +27 -1
  49. package/i18n/en.json +4 -0
  50. package/i18n/sl.json +4 -0
  51. package/package.json +1 -1
  52. package/table/api/descriptors/table.descriptor.d.ts +33 -1
  53. package/table/components/filter/filter-form/filter-form.component.d.ts +11 -2
  54. package/table/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.d.ts +21 -1
  55. package/table/components/table/table.component.d.ts +20 -5
  56. package/table/helpers/filters.d.ts +9 -1
  57. package/table/services/table-feature-config.token.d.ts +1 -0
  58. package/tableview/api/editor/descriptors/field-many.descriptor.d.ts +6 -4
  59. package/tableview/api/editor/descriptors/field.descriptor.d.ts +10 -1
  60. package/version-info.json +5 -5
@@ -1,30 +1,51 @@
1
- import { NgTemplateOutlet } from '@angular/common';
2
- import { ChangeDetectionStrategy, Component, computed, contentChildren, inject, input, signal, viewChild } from '@angular/core';
1
+ import { NgClass, NgStyle, NgTemplateOutlet } from '@angular/common';
2
+ import { ChangeDetectionStrategy, Component, booleanAttribute, computed, contentChildren, inject, input, signal, viewChild } from '@angular/core';
3
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
4
  import { TranslateModule } from '@ngx-translate/core';
4
5
  import { Button } from 'primeng/button';
6
+ import { CheckboxModule } from 'primeng/checkbox';
7
+ import { InputGroupModule } from 'primeng/inputgroup';
8
+ import { InputGroupAddonModule } from 'primeng/inputgroupaddon';
9
+ import { InputTextModule } from 'primeng/inputtext';
5
10
  import { OverlayPanel, OverlayPanelModule } from 'primeng/overlaypanel';
11
+ import { PaginatorModule } from 'primeng/paginator';
6
12
  import { Table } from 'primeng/table';
7
13
  import { TagModule } from 'primeng/tag';
14
+ import { ToggleButtonModule } from 'primeng/togglebutton';
15
+ import { BehaviorSubject, combineLatest, debounceTime, distinctUntilChanged, skip } from 'rxjs';
8
16
  import { COMMONS_MODULE_CONFIG_IT, FilterMatchMode, TemplateDirective, findTemplateByName } from '@mediusinc/mng-commons/core';
9
17
  import { filterApplySerializationConfigToCmp, filterGetDateConfig, filterSetMetadataOnChange, filterSetMetadataOnDisplayChange } from '../../../helpers/filters';
10
18
  import { DataListService } from '../../../services/data-list.service';
19
+ import { COMMONS_TABLE_FEATURE_CONFIG_IT } from '../../../services/table-feature-config.token';
11
20
  import { FilterActiveTagComponent } from '../filter-active-tag/filter-active-tag.component';
12
21
  import { FilterFormComponent } from '../filter-form/filter-form.component';
13
22
  import * as i0 from "@angular/core";
14
23
  import * as i1 from "primeng/overlaypanel";
24
+ import * as i2 from "@ngx-translate/core";
25
+ import * as i3 from "primeng/inputtext";
26
+ import * as i4 from "@angular/forms";
27
+ import * as i5 from "primeng/togglebutton";
15
28
  export class FilterOverlayWithTagComponent {
29
+ static { this.SEARCH_WORD_LIMIT_MAX = 4; }
16
30
  constructor() {
17
31
  this.primeTable = inject(Table);
18
32
  this.dataListService = inject((DataListService));
19
33
  this.moduleConfig = inject(COMMONS_MODULE_CONFIG_IT, { optional: true });
34
+ this.tableConfig = inject(COMMONS_TABLE_FEATURE_CONFIG_IT, { optional: true });
20
35
  this.descriptors = input.required();
21
36
  this.model = input.required();
37
+ this.enableSearch = input(false, { transform: booleanAttribute });
38
+ this.search = input();
39
+ this.searchLimitWordMax = input(undefined);
40
+ this.enableCaseSensitive = input(true, { transform: booleanAttribute });
22
41
  this.filterOverlay = viewChild.required(OverlayPanel);
42
+ this.filterFormComponent = viewChild(FilterFormComponent);
23
43
  this.templates = contentChildren(TemplateDirective);
24
44
  this.titleTemplate = computed(() => findTemplateByName([...this.templates()], 'title'));
25
45
  this.filters = computed(() => {
26
46
  return this.setInitialFilters(this.dataListService.filterMeta());
27
47
  });
48
+ this.showFilterButton = computed(() => this.descriptors().length > 0);
28
49
  this.activeFilters = computed(() => {
29
50
  const filterEntry = this.filters();
30
51
  return this.descriptors()
@@ -42,19 +63,41 @@ export class FilterOverlayWithTagComponent {
42
63
  this.activeFilterProperties = computed(() => this.activeFilters().map(f => f.descriptor.property));
43
64
  this.filterEdit = signal(undefined);
44
65
  this.serializationCfg = {};
66
+ // global search model
67
+ this.searchValue = computed(() => this.search()?.value);
68
+ this.searchCaseSensitive = computed(() => this.search()?.caseSensitive);
69
+ this.searchWordLimitMaxError = signal(false);
70
+ this.searchWordLimitMaxCmp = computed(() => {
71
+ const inputLimit = this.searchLimitWordMax();
72
+ if (inputLimit !== undefined) {
73
+ return inputLimit;
74
+ }
75
+ return this.tableConfig?.searchLimitMaxWords !== undefined ? this.tableConfig?.searchLimitMaxWords : FilterOverlayWithTagComponent.SEARCH_WORD_LIMIT_MAX;
76
+ });
77
+ this.searchValueSubject = new BehaviorSubject(undefined);
78
+ this.searchCaseSensitiveSubject = new BehaviorSubject(undefined);
45
79
  // UI utilities
46
80
  this.clickedOnFilterEvent = undefined;
47
81
  filterApplySerializationConfigToCmp(this.moduleConfig, this.serializationCfg);
82
+ combineLatest([this.searchValueSubject.pipe(debounceTime(500), distinctUntilChanged()), this.searchCaseSensitiveSubject])
83
+ // NOTE: skip is used to ignore propagating the initial value of the subjects
84
+ .pipe(skip(1), takeUntilDestroyed())
85
+ .subscribe(([value, caseSensitive]) => {
86
+ this.updatePrimeTableGlobalFilter(value == null ? this.search()?.value ?? '' : value, caseSensitive == null ? this.search()?.caseSensitive ?? undefined : caseSensitive);
87
+ });
48
88
  }
49
89
  onFilterFormApply(event) {
50
90
  const descriptor = this.descriptors().find(d => d.property === event.property);
51
91
  if (!descriptor) {
52
92
  return;
53
93
  }
54
- this.updatePrimeTableFilter({
94
+ const filterData = {
55
95
  ...filterSetMetadataOnDisplayChange({ caseSensitive: event.caseSensitive }, event.matchMode, event.value, descriptor.filterType, filterGetDateConfig(descriptor, this.serializationCfg)),
56
96
  descriptor: descriptor
57
- });
97
+ };
98
+ this.updatePrimeTableFilter(filterData);
99
+ // this keeps the UI updated properly before closing the overlay
100
+ this.filterEdit.set(filterData);
58
101
  // Resetting the form is done in the onOverlayHide callback because resetting the
59
102
  // values here renders empty inputs for a split second before closing the overlay
60
103
  this.filterOverlay().hide();
@@ -74,11 +117,32 @@ export class FilterOverlayWithTagComponent {
74
117
  onFilterAdd(event) {
75
118
  this.onFilterAddOrEdit(undefined, event);
76
119
  }
120
+ onSearchChange(value) {
121
+ const searchWordLimitMax = this.searchWordLimitMaxCmp();
122
+ if (searchWordLimitMax !== null) {
123
+ const wordCount = value.split(/\s+/).filter(word => word.length > 0).length;
124
+ this.searchWordLimitMaxError.set(wordCount > searchWordLimitMax);
125
+ }
126
+ else {
127
+ this.searchWordLimitMaxError.set(false);
128
+ }
129
+ if (!this.searchWordLimitMaxError()) {
130
+ this.searchValueSubject.next(value);
131
+ }
132
+ }
133
+ onSearchCaseSensitiveChange(value) {
134
+ this.searchCaseSensitiveSubject.next(value);
135
+ }
77
136
  onOverlayHide() {
137
+ // resetting the form is done here for a smoother animation
138
+ if (this.filterFormComponent !== undefined) {
139
+ this.filterFormComponent()?.resetForm();
140
+ }
78
141
  if (this.clickedOnFilterEvent) {
79
142
  // show the overlay again if the user clicked another filter
80
143
  setTimeout(() => {
81
- this.filterOverlay().show(this.clickedOnFilterEvent);
144
+ this.filterEdit.set(this.clickedOnFilterEvent?.filter);
145
+ this.filterOverlay().show(this.clickedOnFilterEvent?.event);
82
146
  this.clickedOnFilterEvent = undefined;
83
147
  }, 0);
84
148
  return;
@@ -92,17 +156,16 @@ export class FilterOverlayWithTagComponent {
92
156
  // If user clicks on the same filter or add that is already open, close the overlay
93
157
  if (this.filterOverlay().overlayVisible && this.filterEdit()?.descriptor.property === filter?.descriptor.property) {
94
158
  this.filterOverlay().hide();
95
- this.filterEdit.set(undefined);
96
159
  return;
97
160
  }
98
- this.filterEdit.set(filter);
99
161
  if (this.filterOverlay().overlayVisible) {
100
162
  // In case the user clicks another filter, 're-showing' the overlay in
101
163
  // the new position is done in the onOverlayHide callback
102
- this.clickedOnFilterEvent = event;
164
+ this.clickedOnFilterEvent = { event, filter };
103
165
  this.filterOverlay().hide();
104
166
  }
105
167
  else {
168
+ this.filterEdit.set(filter);
106
169
  this.filterOverlay().show(event);
107
170
  }
108
171
  }
@@ -125,11 +188,32 @@ export class FilterOverlayWithTagComponent {
125
188
  this.primeTable.filters[filter.descriptor.property] = { ...filter };
126
189
  this.primeTable._filter();
127
190
  }
191
+ updatePrimeTableGlobalFilter(searchQuery, caseSensitive = false) {
192
+ // match mode needs to be set for inline search
193
+ this.primeTable.filters['global'] = { ...{ value: searchQuery, matchMode: 'contains', caseSensitive: caseSensitive } };
194
+ this.primeTable._filter();
195
+ }
128
196
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: FilterOverlayWithTagComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
129
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.5", type: FilterOverlayWithTagComponent, isStandalone: true, selector: "mng-filter-overlay-with-tag", inputs: { descriptors: { classPropertyName: "descriptors", publicName: "descriptors", isSignal: true, isRequired: true, transformFunction: null }, model: { classPropertyName: "model", publicName: "model", isSignal: true, isRequired: true, transformFunction: null } }, queries: [{ propertyName: "templates", predicate: TemplateDirective, isSignal: true }], viewQueries: [{ propertyName: "filterOverlay", first: true, predicate: OverlayPanel, descendants: true, isSignal: true }], ngImport: i0, template: "<div class=\"flex align-items-center flex-grow-1\">\n @if (titleTemplate()) {\n <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n }\n <p-button\n icon=\"pi pi-filter\"\n rounded=\"true\"\n [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n size=\"small\"\n (click)=\"onFilterAdd($event)\"\n [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n </p-button>\n</div>\n@if (activeFilters().length > 0) {\n <div class=\"flex align-items-center flex-wrap mt-2\">\n @for (filter of activeFilters(); track filter.descriptor.property) {\n <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n }\n </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n <mng-filter-form\n [model]=\"model()\"\n [descriptors]=\"descriptors()\"\n [filter]=\"filterEdit()\"\n [disabledProperties]=\"activeFilterProperties()\"\n [overlay]=\"op\"\n (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n", dependencies: [{ kind: "ngmodule", type: OverlayPanelModule }, { kind: "component", type: i1.OverlayPanel, selector: "p-overlayPanel", inputs: ["ariaLabel", "ariaLabelledBy", "dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "ngmodule", type: TagModule }, { kind: "component", type: FilterFormComponent, selector: "mng-filter-form", inputs: ["title", "descriptors", "model", "filter", "overlay", "disabledProperties"], outputs: ["apply"] }, { kind: "component", type: Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "raised", "rounded", "text", "plain", "severity", "outlined", "link", "tabindex", "size", "style", "styleClass", "badgeClass", "ariaLabel", "autofocus"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: FilterActiveTagComponent, selector: "mng-filter-active-tag", inputs: ["model", "filter"], outputs: ["edit", "remove"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
197
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.5", type: FilterOverlayWithTagComponent, isStandalone: true, selector: "mng-filter-overlay-with-tag", inputs: { descriptors: { classPropertyName: "descriptors", publicName: "descriptors", isSignal: true, isRequired: true, transformFunction: null }, model: { classPropertyName: "model", publicName: "model", isSignal: true, isRequired: true, transformFunction: null }, enableSearch: { classPropertyName: "enableSearch", publicName: "enableSearch", isSignal: true, isRequired: false, transformFunction: null }, search: { classPropertyName: "search", publicName: "search", isSignal: true, isRequired: false, transformFunction: null }, searchLimitWordMax: { classPropertyName: "searchLimitWordMax", publicName: "searchLimitWordMax", isSignal: true, isRequired: false, transformFunction: null }, enableCaseSensitive: { classPropertyName: "enableCaseSensitive", publicName: "enableCaseSensitive", isSignal: true, isRequired: false, transformFunction: null } }, queries: [{ propertyName: "templates", predicate: TemplateDirective, isSignal: true }], viewQueries: [{ propertyName: "filterOverlay", first: true, predicate: OverlayPanel, descendants: true, isSignal: true }, { propertyName: "filterFormComponent", first: true, predicate: FilterFormComponent, descendants: true, isSignal: true }], ngImport: i0, template: "<div class=\"flex align-items-center flex-grow-1\">\n @if (titleTemplate()) {\n <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n }\n @if (showFilterButton()) {\n <p-button\n icon=\"pi pi-filter\"\n rounded=\"true\"\n [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n size=\"small\"\n (click)=\"onFilterAdd($event)\"\n [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n </p-button>\n }\n @if (enableSearch()) {\n <div class=\"col-4\" [ngClass]=\"{'ml-3': !showFilterButton() && titleTemplate()}\">\n <span class=\"p-input-icon-left w-full\">\n <i class=\"pi pi-search\"></i>\n <input\n [ngModel]=\"searchValue()\"\n (ngModelChange)=\"onSearchChange($event)\"\n name=\"globalSearch\"\n type=\"text\"\n pInputText\n class=\"search-input w-full\"\n [ngClass]=\"{'ng-invalid': searchWordLimitMaxError()}\"\n [attr.placeholder]=\"'mngFilter.search.input' | translate\"\n [placeholder]=\"'mngFilter.search.input' | translate\" />\n @if (enableCaseSensitive()) {\n <span class=\"toggle-button-input\">\n <p-toggleButton\n styleClass=\"ml-1 p-1 p-button-sm\"\n [ngModel]=\"searchCaseSensitive()\"\n (ngModelChange)=\"onSearchCaseSensitiveChange($event)\"\n onLabel=\"Aa\"\n offLabel=\"Aa\" />\n </span>\n }\n </span>\n @if (searchWordLimitMaxError()) {\n <small class=\"p-error block\">\n {{ 'mngFilter.search.maxWordsExceededMessage' | translate }}\n </small>\n }\n </div>\n }\n</div>\n@if (activeFilters().length > 0) {\n <div class=\"flex align-items-center flex-wrap mt-2\">\n @for (filter of activeFilters(); track filter.descriptor.property) {\n <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n }\n </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n <mng-filter-form\n [model]=\"model()\"\n [descriptors]=\"descriptors()\"\n [filter]=\"filterEdit()\"\n [disabledProperties]=\"activeFilterProperties()\"\n [overlay]=\"op\"\n (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n", styles: [".toggle-button-input{position:absolute;right:.35rem;top:50%;transform:translateY(-50%)}\n"], dependencies: [{ kind: "ngmodule", type: OverlayPanelModule }, { kind: "component", type: i1.OverlayPanel, selector: "p-overlayPanel", inputs: ["ariaLabel", "ariaLabelledBy", "dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: TagModule }, { kind: "component", type: FilterFormComponent, selector: "mng-filter-form", inputs: ["title", "descriptors", "model", "filter", "overlay", "disabledProperties"], outputs: ["apply"] }, { kind: "component", type: Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "raised", "rounded", "text", "plain", "severity", "outlined", "link", "tabindex", "size", "style", "styleClass", "badgeClass", "ariaLabel", "autofocus"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: FilterActiveTagComponent, selector: "mng-filter-active-tag", inputs: ["model", "filter"], outputs: ["edit", "remove"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: InputTextModule }, { kind: "directive", type: i3.InputText, selector: "[pInputText]", inputs: ["variant"] }, { kind: "ngmodule", type: PaginatorModule }, { kind: "directive", type: i4.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: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "ngmodule", type: ToggleButtonModule }, { kind: "component", type: i5.ToggleButton, selector: "p-toggleButton", inputs: ["onLabel", "offLabel", "onIcon", "offIcon", "ariaLabel", "ariaLabelledBy", "disabled", "style", "styleClass", "inputId", "tabindex", "iconPos", "autofocus"], outputs: ["onChange"] }, { kind: "ngmodule", type: InputGroupModule }, { kind: "ngmodule", type: InputGroupAddonModule }, { kind: "ngmodule", type: CheckboxModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
130
198
  }
131
199
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: FilterOverlayWithTagComponent, decorators: [{
132
200
  type: Component,
133
- args: [{ standalone: true, selector: 'mng-filter-overlay-with-tag', imports: [OverlayPanelModule, TranslateModule, TagModule, FilterFormComponent, Button, FilterActiveTagComponent, NgTemplateOutlet], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex align-items-center flex-grow-1\">\n @if (titleTemplate()) {\n <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n }\n <p-button\n icon=\"pi pi-filter\"\n rounded=\"true\"\n [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n size=\"small\"\n (click)=\"onFilterAdd($event)\"\n [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n </p-button>\n</div>\n@if (activeFilters().length > 0) {\n <div class=\"flex align-items-center flex-wrap mt-2\">\n @for (filter of activeFilters(); track filter.descriptor.property) {\n <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n }\n </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n <mng-filter-form\n [model]=\"model()\"\n [descriptors]=\"descriptors()\"\n [filter]=\"filterEdit()\"\n [disabledProperties]=\"activeFilterProperties()\"\n [overlay]=\"op\"\n (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n" }]
201
+ args: [{ standalone: true, selector: 'mng-filter-overlay-with-tag', imports: [
202
+ OverlayPanelModule,
203
+ TranslateModule,
204
+ TagModule,
205
+ FilterFormComponent,
206
+ Button,
207
+ FilterActiveTagComponent,
208
+ NgTemplateOutlet,
209
+ InputTextModule,
210
+ PaginatorModule,
211
+ NgClass,
212
+ ToggleButtonModule,
213
+ InputGroupModule,
214
+ InputGroupAddonModule,
215
+ CheckboxModule,
216
+ NgStyle
217
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex align-items-center flex-grow-1\">\n @if (titleTemplate()) {\n <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n }\n @if (showFilterButton()) {\n <p-button\n icon=\"pi pi-filter\"\n rounded=\"true\"\n [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n size=\"small\"\n (click)=\"onFilterAdd($event)\"\n [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n </p-button>\n }\n @if (enableSearch()) {\n <div class=\"col-4\" [ngClass]=\"{'ml-3': !showFilterButton() && titleTemplate()}\">\n <span class=\"p-input-icon-left w-full\">\n <i class=\"pi pi-search\"></i>\n <input\n [ngModel]=\"searchValue()\"\n (ngModelChange)=\"onSearchChange($event)\"\n name=\"globalSearch\"\n type=\"text\"\n pInputText\n class=\"search-input w-full\"\n [ngClass]=\"{'ng-invalid': searchWordLimitMaxError()}\"\n [attr.placeholder]=\"'mngFilter.search.input' | translate\"\n [placeholder]=\"'mngFilter.search.input' | translate\" />\n @if (enableCaseSensitive()) {\n <span class=\"toggle-button-input\">\n <p-toggleButton\n styleClass=\"ml-1 p-1 p-button-sm\"\n [ngModel]=\"searchCaseSensitive()\"\n (ngModelChange)=\"onSearchCaseSensitiveChange($event)\"\n onLabel=\"Aa\"\n offLabel=\"Aa\" />\n </span>\n }\n </span>\n @if (searchWordLimitMaxError()) {\n <small class=\"p-error block\">\n {{ 'mngFilter.search.maxWordsExceededMessage' | translate }}\n </small>\n }\n </div>\n }\n</div>\n@if (activeFilters().length > 0) {\n <div class=\"flex align-items-center flex-wrap mt-2\">\n @for (filter of activeFilters(); track filter.descriptor.property) {\n <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n }\n </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n <mng-filter-form\n [model]=\"model()\"\n [descriptors]=\"descriptors()\"\n [filter]=\"filterEdit()\"\n [disabledProperties]=\"activeFilterProperties()\"\n [overlay]=\"op\"\n (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n", styles: [".toggle-button-input{position:absolute;right:.35rem;top:50%;transform:translateY(-50%)}\n"] }]
134
218
  }], ctorParameters: () => [] });
135
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-overlay-with-tag.component.js","sourceRoot":"","sources":["../../../../../../table/src/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.ts","../../../../../../table/src/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.html"],"names":[],"mappings":"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","sourcesContent":["import {NgTemplateOutlet} from '@angular/common';\nimport {ChangeDetectionStrategy, Component, Signal, TemplateRef, computed, contentChildren, inject, input, signal, viewChild} from '@angular/core';\n\nimport {TranslateModule} from '@ngx-translate/core';\nimport {Button} from 'primeng/button';\nimport {OverlayPanel, OverlayPanelModule} from 'primeng/overlaypanel';\nimport {Table} from 'primeng/table';\nimport {TagModule} from 'primeng/tag';\n\nimport {COMMONS_MODULE_CONFIG_IT, CommonsFilterMetadata, CommonsSerializationConfig, FilterMatchMode, TemplateDirective, findTemplateByName} from '@mediusinc/mng-commons/core';\nimport {FilterDescriptor} from '@mediusinc/mng-commons/filter';\nimport {ModelDescriptor} from '@mediusinc/mng-commons/model';\n\nimport {filterApplySerializationConfigToCmp, filterGetDateConfig, filterSetMetadataOnChange, filterSetMetadataOnDisplayChange} from '../../../helpers/filters';\nimport {CommonsFilterMetadataWithDescriptor, FilterActiveTagEvent, FilterFormEmitEvent} from '../../../models/filter.model';\nimport {DataListService} from '../../../services/data-list.service';\nimport {FilterActiveTagComponent} from '../filter-active-tag/filter-active-tag.component';\nimport {FilterFormComponent} from '../filter-form/filter-form.component';\n\n@Component({\n    standalone: true,\n    selector: 'mng-filter-overlay-with-tag',\n    templateUrl: './filter-overlay-with-tag.component.html',\n    imports: [OverlayPanelModule, TranslateModule, TagModule, FilterFormComponent, Button, FilterActiveTagComponent, NgTemplateOutlet],\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class FilterOverlayWithTagComponent {\n    private readonly primeTable = inject(Table);\n    private readonly dataListService = inject(DataListService<any>);\n    private readonly moduleConfig = inject(COMMONS_MODULE_CONFIG_IT, {optional: true});\n\n    public descriptors = input.required<FilterDescriptor<any, any>[]>();\n    public model = input.required<ModelDescriptor<any>>();\n\n    private filterOverlay = viewChild.required(OverlayPanel);\n\n    public templates = contentChildren(TemplateDirective);\n    public titleTemplate: Signal<TemplateRef<any> | null> = computed(() => findTemplateByName([...this.templates()], 'title'));\n\n    public filters = computed(() => {\n        return this.setInitialFilters(this.dataListService.filterMeta());\n    });\n\n    public activeFilters = computed(() => {\n        const filterEntry = this.filters();\n        return this.descriptors()\n            .filter(d => {\n                const filter = filterEntry[d.property];\n                return (\n                    filter &&\n                    filter.matchMode !== undefined &&\n                    filter.displayMatchMode !== undefined &&\n                    (filter.matchMode === FilterMatchMode.Exists ||\n                        filter.matchMode === FilterMatchMode.DoesNotExist ||\n                        (filter.value !== undefined && filter.displayValue !== undefined))\n                );\n            })\n            .map(d => filterEntry[d.property]!);\n    });\n    public activeFilterProperties = computed(() => this.activeFilters().map(f => f.descriptor.property));\n    public filterEdit = signal<CommonsFilterMetadataWithDescriptor | undefined>(undefined);\n\n    private serializationCfg: CommonsSerializationConfig = {};\n\n    // UI utilities\n    private clickedOnFilterEvent: Event | undefined = undefined;\n\n    constructor() {\n        filterApplySerializationConfigToCmp(this.moduleConfig, this.serializationCfg);\n    }\n\n    onFilterFormApply(event: FilterFormEmitEvent) {\n        const descriptor = this.descriptors().find(d => d.property === event.property);\n        if (!descriptor) {\n            return;\n        }\n\n        this.updatePrimeTableFilter({\n            ...filterSetMetadataOnDisplayChange(\n                {caseSensitive: event.caseSensitive},\n                event.matchMode,\n                event.value,\n                descriptor.filterType,\n                filterGetDateConfig(descriptor, this.serializationCfg)\n            ),\n            descriptor: descriptor\n        });\n\n        // Resetting the form is done in the onOverlayHide callback because resetting the\n        // values here renders empty inputs for a split second before closing the overlay\n        this.filterOverlay().hide();\n    }\n\n    onFilterRemove(event: FilterActiveTagEvent) {\n        this.updatePrimeTableFilter({\n            ...this.filters()[event.filter.descriptor.property],\n            matchMode: undefined,\n            displayMatchMode: undefined,\n            value: undefined,\n            displayValue: undefined\n        });\n    }\n\n    onFilterEdit(event: FilterActiveTagEvent) {\n        this.onFilterAddOrEdit(event.filter, event.event);\n    }\n\n    onFilterAdd(event: Event) {\n        this.onFilterAddOrEdit(undefined, event);\n    }\n\n    onOverlayHide() {\n        if (this.clickedOnFilterEvent) {\n            // show the overlay again if the user clicked another filter\n            setTimeout(() => {\n                this.filterOverlay().show(this.clickedOnFilterEvent);\n                this.clickedOnFilterEvent = undefined;\n            }, 0);\n            return;\n        } else {\n            // clean state\n            this.filterEdit.set(undefined);\n        }\n    }\n\n    private onFilterAddOrEdit(filter?: CommonsFilterMetadataWithDescriptor, event?: Event) {\n        // If user clicks on the same filter or add that is already open, close the overlay\n        if (this.filterOverlay().overlayVisible && this.filterEdit()?.descriptor.property === filter?.descriptor.property) {\n            this.filterOverlay().hide();\n            this.filterEdit.set(undefined);\n            return;\n        }\n\n        this.filterEdit.set(filter);\n        if (this.filterOverlay().overlayVisible) {\n            // In case the user clicks another filter, 're-showing' the overlay in\n            // the new position is done in the onOverlayHide callback\n            this.clickedOnFilterEvent = event;\n            this.filterOverlay().hide();\n        } else {\n            this.filterOverlay().show(event);\n        }\n    }\n\n    private setInitialFilters(filtersInit: Record<string, CommonsFilterMetadata>) {\n        const filtersResult: {[p: string]: CommonsFilterMetadataWithDescriptor} = {};\n\n        for (const property in filtersInit) {\n            const filterInit = filtersInit[property];\n\n            const descriptor = this.descriptors().find(filterDescriptor => filterDescriptor.property === property);\n            if (!descriptor) {\n                continue;\n            }\n\n            filtersResult[property] = {\n                ...filterSetMetadataOnChange(filterInit, filterInit.matchMode, filterInit.value, descriptor.filterType, filterGetDateConfig(descriptor, this.serializationCfg)),\n                descriptor: descriptor\n            };\n        }\n\n        return filtersResult;\n    }\n\n    private updatePrimeTableFilter(filter: CommonsFilterMetadataWithDescriptor) {\n        this.primeTable.filters[filter.descriptor.property] = {...filter};\n        this.primeTable._filter();\n    }\n}\n","<div class=\"flex align-items-center flex-grow-1\">\n    @if (titleTemplate()) {\n        <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n    }\n    <p-button\n        icon=\"pi pi-filter\"\n        rounded=\"true\"\n        [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n        size=\"small\"\n        (click)=\"onFilterAdd($event)\"\n        [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n    </p-button>\n</div>\n@if (activeFilters().length > 0) {\n    <div class=\"flex align-items-center flex-wrap mt-2\">\n        @for (filter of activeFilters(); track filter.descriptor.property) {\n            <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n        }\n    </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n    <mng-filter-form\n        [model]=\"model()\"\n        [descriptors]=\"descriptors()\"\n        [filter]=\"filterEdit()\"\n        [disabledProperties]=\"activeFilterProperties()\"\n        [overlay]=\"op\"\n        (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n"]}
219
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-overlay-with-tag.component.js","sourceRoot":"","sources":["../../../../../../table/src/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.ts","../../../../../../table/src/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.html"],"names":[],"mappings":"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","sourcesContent":["import {NgClass, NgStyle, NgTemplateOutlet} from '@angular/common';\nimport {ChangeDetectionStrategy, Component, Signal, TemplateRef, booleanAttribute, computed, contentChildren, inject, input, signal, viewChild} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\nimport {TranslateModule} from '@ngx-translate/core';\nimport {Button} from 'primeng/button';\nimport {CheckboxModule} from 'primeng/checkbox';\nimport {InputGroupModule} from 'primeng/inputgroup';\nimport {InputGroupAddonModule} from 'primeng/inputgroupaddon';\nimport {InputTextModule} from 'primeng/inputtext';\nimport {OverlayPanel, OverlayPanelModule} from 'primeng/overlaypanel';\nimport {PaginatorModule} from 'primeng/paginator';\nimport {Table} from 'primeng/table';\nimport {TagModule} from 'primeng/tag';\nimport {ToggleButtonModule} from 'primeng/togglebutton';\nimport {BehaviorSubject, combineLatest, debounceTime, distinctUntilChanged, skip} from 'rxjs';\n\nimport {COMMONS_MODULE_CONFIG_IT, CommonsFilterMetadata, CommonsSerializationConfig, FilterMatchMode, TemplateDirective, findTemplateByName} from '@mediusinc/mng-commons/core';\nimport {FilterDescriptor} from '@mediusinc/mng-commons/filter';\nimport {ModelDescriptor} from '@mediusinc/mng-commons/model';\n\nimport {filterApplySerializationConfigToCmp, filterGetDateConfig, filterSetMetadataOnChange, filterSetMetadataOnDisplayChange} from '../../../helpers/filters';\nimport {CommonsFilterMetadataWithDescriptor, FilterActiveTagEvent, FilterFormEmitEvent} from '../../../models/filter.model';\nimport {DataListService} from '../../../services/data-list.service';\nimport {COMMONS_TABLE_FEATURE_CONFIG_IT} from '../../../services/table-feature-config.token';\nimport {FilterActiveTagComponent} from '../filter-active-tag/filter-active-tag.component';\nimport {FilterFormComponent} from '../filter-form/filter-form.component';\n\n@Component({\n    standalone: true,\n    selector: 'mng-filter-overlay-with-tag',\n    templateUrl: './filter-overlay-with-tag.component.html',\n    styleUrls: ['./filter-overlay-with-tag.component.scss'],\n    imports: [\n        OverlayPanelModule,\n        TranslateModule,\n        TagModule,\n        FilterFormComponent,\n        Button,\n        FilterActiveTagComponent,\n        NgTemplateOutlet,\n        InputTextModule,\n        PaginatorModule,\n        NgClass,\n        ToggleButtonModule,\n        InputGroupModule,\n        InputGroupAddonModule,\n        CheckboxModule,\n        NgStyle\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class FilterOverlayWithTagComponent {\n    private static readonly SEARCH_WORD_LIMIT_MAX = 4;\n\n    private readonly primeTable = inject(Table);\n    private readonly dataListService = inject(DataListService<any>);\n    private readonly moduleConfig = inject(COMMONS_MODULE_CONFIG_IT, {optional: true});\n    private readonly tableConfig = inject(COMMONS_TABLE_FEATURE_CONFIG_IT, {optional: true});\n\n    public descriptors = input.required<FilterDescriptor<any, any>[]>();\n    public model = input.required<ModelDescriptor<any>>();\n    public enableSearch = input(false, {transform: booleanAttribute});\n    public search = input<{value: string; caseSensitive: boolean} | undefined>();\n    public searchLimitWordMax = input<number | undefined | null>(undefined);\n    public enableCaseSensitive = input(true, {transform: booleanAttribute});\n\n    private filterOverlay = viewChild.required(OverlayPanel);\n    private filterFormComponent = viewChild(FilterFormComponent);\n\n    public templates = contentChildren(TemplateDirective);\n    public titleTemplate: Signal<TemplateRef<any> | null> = computed(() => findTemplateByName([...this.templates()], 'title'));\n\n    public filters = computed(() => {\n        return this.setInitialFilters(this.dataListService.filterMeta());\n    });\n\n    public showFilterButton = computed(() => this.descriptors().length > 0);\n\n    public activeFilters = computed(() => {\n        const filterEntry = this.filters();\n        return this.descriptors()\n            .filter(d => {\n                const filter = filterEntry[d.property];\n                return (\n                    filter &&\n                    filter.matchMode !== undefined &&\n                    filter.displayMatchMode !== undefined &&\n                    (filter.matchMode === FilterMatchMode.Exists ||\n                        filter.matchMode === FilterMatchMode.DoesNotExist ||\n                        (filter.value !== undefined && filter.displayValue !== undefined))\n                );\n            })\n            .map(d => filterEntry[d.property]!);\n    });\n    public activeFilterProperties = computed(() => this.activeFilters().map(f => f.descriptor.property));\n    public filterEdit = signal<CommonsFilterMetadataWithDescriptor | undefined>(undefined);\n\n    private serializationCfg: CommonsSerializationConfig = {};\n\n    // global search model\n    public searchValue = computed(() => this.search()?.value);\n    public searchCaseSensitive = computed(() => this.search()?.caseSensitive);\n    public searchWordLimitMaxError = signal(false);\n    private searchWordLimitMaxCmp = computed(() => {\n        const inputLimit = this.searchLimitWordMax();\n        if (inputLimit !== undefined) {\n            return inputLimit;\n        }\n        return this.tableConfig?.searchLimitMaxWords !== undefined ? this.tableConfig?.searchLimitMaxWords : FilterOverlayWithTagComponent.SEARCH_WORD_LIMIT_MAX;\n    });\n    private searchValueSubject = new BehaviorSubject<string | undefined>(undefined);\n    private searchCaseSensitiveSubject = new BehaviorSubject<boolean | undefined>(undefined);\n\n    // UI utilities\n    private clickedOnFilterEvent: {event?: Event; filter?: CommonsFilterMetadataWithDescriptor} | undefined = undefined;\n\n    constructor() {\n        filterApplySerializationConfigToCmp(this.moduleConfig, this.serializationCfg);\n\n        combineLatest([this.searchValueSubject.pipe(debounceTime(500), distinctUntilChanged()), this.searchCaseSensitiveSubject])\n            // NOTE: skip is used to ignore propagating the initial value of the subjects\n            .pipe(skip(1), takeUntilDestroyed())\n            .subscribe(([value, caseSensitive]) => {\n                this.updatePrimeTableGlobalFilter(\n                    value == null ? this.search()?.value ?? '' : value,\n                    caseSensitive == null ? this.search()?.caseSensitive ?? undefined : caseSensitive\n                );\n            });\n    }\n\n    onFilterFormApply(event: FilterFormEmitEvent) {\n        const descriptor = this.descriptors().find(d => d.property === event.property);\n        if (!descriptor) {\n            return;\n        }\n\n        const filterData = {\n            ...filterSetMetadataOnDisplayChange(\n                {caseSensitive: event.caseSensitive},\n                event.matchMode,\n                event.value,\n                descriptor.filterType,\n                filterGetDateConfig(descriptor, this.serializationCfg)\n            ),\n            descriptor: descriptor\n        };\n\n        this.updatePrimeTableFilter(filterData);\n\n        // this keeps the UI updated properly before closing the overlay\n        this.filterEdit.set(filterData);\n\n        // Resetting the form is done in the onOverlayHide callback because resetting the\n        // values here renders empty inputs for a split second before closing the overlay\n        this.filterOverlay().hide();\n    }\n\n    onFilterRemove(event: FilterActiveTagEvent) {\n        this.updatePrimeTableFilter({\n            ...this.filters()[event.filter.descriptor.property],\n            matchMode: undefined,\n            displayMatchMode: undefined,\n            value: undefined,\n            displayValue: undefined\n        });\n    }\n\n    onFilterEdit(event: FilterActiveTagEvent) {\n        this.onFilterAddOrEdit(event.filter, event.event);\n    }\n\n    onFilterAdd(event: Event) {\n        this.onFilterAddOrEdit(undefined, event);\n    }\n\n    onSearchChange(value: string): void {\n        const searchWordLimitMax = this.searchWordLimitMaxCmp();\n        if (searchWordLimitMax !== null) {\n            const wordCount = value.split(/\\s+/).filter(word => word.length > 0).length;\n            this.searchWordLimitMaxError.set(wordCount > searchWordLimitMax);\n        } else {\n            this.searchWordLimitMaxError.set(false);\n        }\n\n        if (!this.searchWordLimitMaxError()) {\n            this.searchValueSubject.next(value);\n        }\n    }\n\n    onSearchCaseSensitiveChange(value: boolean) {\n        this.searchCaseSensitiveSubject.next(value);\n    }\n\n    onOverlayHide() {\n        // resetting the form is done here for a smoother animation\n        if (this.filterFormComponent !== undefined) {\n            this.filterFormComponent()?.resetForm();\n        }\n\n        if (this.clickedOnFilterEvent) {\n            // show the overlay again if the user clicked another filter\n            setTimeout(() => {\n                this.filterEdit.set(this.clickedOnFilterEvent?.filter);\n                this.filterOverlay().show(this.clickedOnFilterEvent?.event);\n                this.clickedOnFilterEvent = undefined;\n            }, 0);\n            return;\n        } else {\n            // clean state\n            this.filterEdit.set(undefined);\n        }\n    }\n\n    private onFilterAddOrEdit(filter?: CommonsFilterMetadataWithDescriptor, event?: Event) {\n        // If user clicks on the same filter or add that is already open, close the overlay\n        if (this.filterOverlay().overlayVisible && this.filterEdit()?.descriptor.property === filter?.descriptor.property) {\n            this.filterOverlay().hide();\n            return;\n        }\n\n        if (this.filterOverlay().overlayVisible) {\n            // In case the user clicks another filter, 're-showing' the overlay in\n            // the new position is done in the onOverlayHide callback\n            this.clickedOnFilterEvent = {event, filter};\n            this.filterOverlay().hide();\n        } else {\n            this.filterEdit.set(filter);\n            this.filterOverlay().show(event);\n        }\n    }\n\n    private setInitialFilters(filtersInit: Record<string, CommonsFilterMetadata>) {\n        const filtersResult: {[p: string]: CommonsFilterMetadataWithDescriptor} = {};\n\n        for (const property in filtersInit) {\n            const filterInit = filtersInit[property];\n\n            const descriptor = this.descriptors().find(filterDescriptor => filterDescriptor.property === property);\n            if (!descriptor) {\n                continue;\n            }\n\n            filtersResult[property] = {\n                ...filterSetMetadataOnChange(filterInit, filterInit.matchMode, filterInit.value, descriptor.filterType, filterGetDateConfig(descriptor, this.serializationCfg)),\n                descriptor: descriptor\n            };\n        }\n\n        return filtersResult;\n    }\n\n    private updatePrimeTableFilter(filter: CommonsFilterMetadataWithDescriptor) {\n        this.primeTable.filters[filter.descriptor.property] = {...filter};\n        this.primeTable._filter();\n    }\n\n    private updatePrimeTableGlobalFilter(searchQuery: string | null, caseSensitive = false) {\n        // match mode needs to be set for inline search\n        this.primeTable.filters['global'] = {...{value: searchQuery, matchMode: 'contains', caseSensitive: caseSensitive}};\n        this.primeTable._filter();\n    }\n}\n","<div class=\"flex align-items-center flex-grow-1\">\n    @if (titleTemplate()) {\n        <ng-container *ngTemplateOutlet=\"titleTemplate()\"></ng-container>\n    }\n    @if (showFilterButton()) {\n        <p-button\n            icon=\"pi pi-filter\"\n            rounded=\"true\"\n            [severity]=\"activeFilters().length > 0 ? 'primary' : 'secondary'\"\n            size=\"small\"\n            (click)=\"onFilterAdd($event)\"\n            [styleClass]=\"'my-1 mr-2' + (titleTemplate() ? ' ml-3' : '')\">\n        </p-button>\n    }\n    @if (enableSearch()) {\n        <div class=\"col-4\" [ngClass]=\"{'ml-3': !showFilterButton() && titleTemplate()}\">\n            <span class=\"p-input-icon-left w-full\">\n                <i class=\"pi pi-search\"></i>\n                <input\n                    [ngModel]=\"searchValue()\"\n                    (ngModelChange)=\"onSearchChange($event)\"\n                    name=\"globalSearch\"\n                    type=\"text\"\n                    pInputText\n                    class=\"search-input w-full\"\n                    [ngClass]=\"{'ng-invalid': searchWordLimitMaxError()}\"\n                    [attr.placeholder]=\"'mngFilter.search.input' | translate\"\n                    [placeholder]=\"'mngFilter.search.input' | translate\" />\n                @if (enableCaseSensitive()) {\n                    <span class=\"toggle-button-input\">\n                        <p-toggleButton\n                            styleClass=\"ml-1 p-1 p-button-sm\"\n                            [ngModel]=\"searchCaseSensitive()\"\n                            (ngModelChange)=\"onSearchCaseSensitiveChange($event)\"\n                            onLabel=\"Aa\"\n                            offLabel=\"Aa\" />\n                    </span>\n                }\n            </span>\n            @if (searchWordLimitMaxError()) {\n                <small class=\"p-error block\">\n                    {{ 'mngFilter.search.maxWordsExceededMessage' | translate }}\n                </small>\n            }\n        </div>\n    }\n</div>\n@if (activeFilters().length > 0) {\n    <div class=\"flex align-items-center flex-wrap mt-2\">\n        @for (filter of activeFilters(); track filter.descriptor.property) {\n            <mng-filter-active-tag [model]=\"model()\" [filter]=\"filter\" (edit)=\"onFilterEdit($event)\" (remove)=\"onFilterRemove($event)\" />\n        }\n    </div>\n}\n<p-overlayPanel #op [showCloseIcon]=\"true\" (onHide)=\"onOverlayHide()\" appendTo=\"body\">\n    <mng-filter-form\n        [model]=\"model()\"\n        [descriptors]=\"descriptors()\"\n        [filter]=\"filterEdit()\"\n        [disabledProperties]=\"activeFilterProperties()\"\n        [overlay]=\"op\"\n        (apply)=\"onFilterFormApply($event)\" />\n</p-overlayPanel>\n"]}