@mediusinc/mng-commons 5.3.0-rc.1 → 5.3.0-rc.3

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 (106) hide show
  1. package/core/data-list/filter-metadata.model.d.ts +20 -0
  2. package/core/helpers/number.d.ts +5 -0
  3. package/core/helpers/type-helpers.d.ts +1 -0
  4. package/core/index.d.ts +2 -0
  5. package/core/provide.d.ts +3 -2
  6. package/core/services/tokens/module-config.token.d.ts +7 -6
  7. package/esm2022/core/data-list/data-list-params-helpers.mjs +3 -1
  8. package/esm2022/core/data-list/filter-metadata.model.mjs +2 -0
  9. package/esm2022/core/helpers/date.mjs +2 -2
  10. package/esm2022/core/helpers/number.mjs +19 -0
  11. package/esm2022/core/helpers/type-helpers.mjs +1 -1
  12. package/esm2022/core/index.mjs +3 -1
  13. package/esm2022/core/pipes/boolean.pipe.mjs +7 -1
  14. package/esm2022/core/provide.mjs +4 -3
  15. package/esm2022/core/services/tokens/module-config.token.mjs +1 -1
  16. package/esm2022/filter/descriptors/filter-lookup.descriptor.mjs +2 -3
  17. package/esm2022/filter/descriptors/filter.descriptor.mjs +76 -31
  18. package/esm2022/filter/models/filter.model.mjs +6 -1
  19. package/esm2022/form/components/date-range/date-range.component.mjs +45 -14
  20. package/esm2022/form/components/dropdown/dropdown.component.mjs +77 -93
  21. package/esm2022/form/components/number-range/number-range.component.mjs +29 -12
  22. package/esm2022/table/api/descriptors/column.descriptor.mjs +2 -1
  23. package/esm2022/table/api/descriptors/table.descriptor.mjs +22 -21
  24. package/esm2022/table/api/index.mjs +2 -1
  25. package/esm2022/table/api/models/row-expandable-component.model.mjs +2 -0
  26. package/esm2022/table/api/models/table-columns.model.mjs +1 -1
  27. package/esm2022/table/api/models/table.model.mjs +1 -1
  28. package/esm2022/table/components/column-filter/column-filter.component.mjs +396 -0
  29. package/esm2022/table/components/column-value/column-value.component.mjs +3 -3
  30. package/esm2022/table/components/filter/filter-active-tag/filter-active-tag.component.mjs +52 -0
  31. package/esm2022/table/components/filter/filter-form/filter-form.component.mjs +197 -0
  32. package/esm2022/table/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.mjs +135 -0
  33. package/esm2022/table/components/table/table.component.mjs +34 -25
  34. package/esm2022/table/helpers/filters.mjs +297 -0
  35. package/esm2022/table/index.mjs +13 -2
  36. package/esm2022/table/models/filter.model.mjs +2 -0
  37. package/esm2022/table/pipes/filter-value.pipe.mjs +77 -0
  38. package/esm2022/table/provide.mjs +19 -0
  39. package/esm2022/table/services/data-list.service.mjs +1 -1
  40. package/esm2022/table/services/table-feature-config.token.mjs +3 -0
  41. package/esm2022/tableview/action/components/action/action.component.mjs +3 -3
  42. package/esm2022/tableview/action/components/editor/injector-context/action-editor-injector-context.component.mjs +2 -2
  43. package/esm2022/tableview/action/components/localization/data-language-dropdown.component.mjs +2 -2
  44. package/esm2022/tableview/action/components/table/action-table.component.mjs +3 -3
  45. package/esm2022/tableview/api/editor/descriptors/field-base.descriptor.mjs +1 -3
  46. package/esm2022/tableview/api/editor/descriptors/field.descriptor.mjs +14 -1
  47. package/esm2022/tableview/api/editor/models/editor-fields.model.mjs +1 -1
  48. package/esm2022/tableview/api/editor/models/field-image-preview.model.mjs +2 -0
  49. package/esm2022/tableview/api/index.mjs +2 -1
  50. package/esm2022/tableview/api/tableview/descriptors/tableview.descriptor.mjs +11 -13
  51. package/esm2022/tableview/editor/components/formly/fields/formly-field-autocomplete/formly-field-autocomplete.component.mjs +3 -3
  52. package/esm2022/tableview/editor/components/formly/fields/formly-field-datepicker/formly-field-datepicker.component.mjs +3 -3
  53. package/esm2022/tableview/editor/components/formly/fields/formly-field-dropdown/formly-field-dropdown.component.mjs +3 -3
  54. package/esm2022/tableview/editor/components/formly/fields/formly-field-input/formly-field-input.component.mjs +9 -3
  55. package/esm2022/tableview/editor/components/formly/fields/formly-field-lookup-dialog/formly-field-lookup-dialog.component.mjs +3 -3
  56. package/esm2022/tableview/editor/components/formly/fields/formly-field-table-dialog-form/formly-field-table-dialog-form.component.mjs +3 -3
  57. package/esm2022/tableview/editor/components/formly/fields/formly-field-table-dialog-multiselect/formly-field-table-dialog-multiselect.component.mjs +3 -3
  58. package/esm2022/tableview/tableview/components/tableview/tableview.component.mjs +3 -4
  59. package/fesm2022/mediusinc-mng-commons-core.mjs +32 -4
  60. package/fesm2022/mediusinc-mng-commons-core.mjs.map +1 -1
  61. package/fesm2022/mediusinc-mng-commons-filter.mjs +81 -32
  62. package/fesm2022/mediusinc-mng-commons-filter.mjs.map +1 -1
  63. package/fesm2022/mediusinc-mng-commons-form.mjs +149 -117
  64. package/fesm2022/mediusinc-mng-commons-form.mjs.map +1 -1
  65. package/fesm2022/mediusinc-mng-commons-table-api.mjs +22 -20
  66. package/fesm2022/mediusinc-mng-commons-table-api.mjs.map +1 -1
  67. package/fesm2022/mediusinc-mng-commons-table.mjs +833 -360
  68. package/fesm2022/mediusinc-mng-commons-table.mjs.map +1 -1
  69. package/fesm2022/mediusinc-mng-commons-tableview-api.mjs +23 -14
  70. package/fesm2022/mediusinc-mng-commons-tableview-api.mjs.map +1 -1
  71. package/fesm2022/mediusinc-mng-commons-tableview.mjs +27 -22
  72. package/fesm2022/mediusinc-mng-commons-tableview.mjs.map +1 -1
  73. package/filter/descriptors/filter.descriptor.d.ts +60 -16
  74. package/filter/models/filter.model.d.ts +4 -0
  75. package/form/components/date-range/date-range.component.d.ts +9 -3
  76. package/form/components/dropdown/dropdown.component.d.ts +6 -7
  77. package/form/components/number-range/number-range.component.d.ts +8 -4
  78. package/i18n/en.json +32 -0
  79. package/i18n/sl.json +32 -0
  80. package/package.json +1 -1
  81. package/table/api/descriptors/table.descriptor.d.ts +15 -8
  82. package/table/api/index.d.ts +1 -0
  83. package/table/api/models/row-expandable-component.model.d.ts +4 -0
  84. package/table/api/models/table-columns.model.d.ts +3 -3
  85. package/table/api/models/table.model.d.ts +0 -12
  86. package/table/components/{column-filter-full/column-filter-full.component.d.ts → column-filter/column-filter.component.d.ts} +23 -38
  87. package/table/components/filter/filter-active-tag/filter-active-tag.component.d.ts +20 -0
  88. package/table/components/filter/filter-form/filter-form.component.d.ts +53 -0
  89. package/table/components/filter/filter-overlay-with-tag/filter-overlay-with-tag.component.d.ts +35 -0
  90. package/table/components/table/table.component.d.ts +10 -7
  91. package/table/helpers/filters.d.ts +31 -0
  92. package/table/index.d.ts +10 -1
  93. package/table/models/filter.model.d.ts +24 -0
  94. package/table/pipes/filter-value.pipe.d.ts +19 -0
  95. package/table/provide.d.ts +8 -0
  96. package/table/services/data-list.service.d.ts +3 -2
  97. package/table/services/table-feature-config.token.d.ts +6 -0
  98. package/tableview/api/editor/descriptors/field-base.descriptor.d.ts +4 -4
  99. package/tableview/api/editor/descriptors/field.descriptor.d.ts +13 -3
  100. package/tableview/api/editor/models/editor-fields.model.d.ts +4 -4
  101. package/tableview/api/editor/models/field-image-preview.model.d.ts +4 -0
  102. package/tableview/api/index.d.ts +1 -0
  103. package/tableview/api/tableview/descriptors/tableview.descriptor.d.ts +8 -6
  104. package/tableview/editor/components/formly/fields/formly-field-input/formly-field-input.component.d.ts +1 -0
  105. package/version-info.json +5 -5
  106. package/esm2022/table/components/column-filter-full/column-filter-full.component.mjs +0 -655
@@ -0,0 +1,135 @@
1
+ import { NgTemplateOutlet } from '@angular/common';
2
+ import { ChangeDetectionStrategy, Component, computed, contentChildren, inject, input, signal, viewChild } from '@angular/core';
3
+ import { TranslateModule } from '@ngx-translate/core';
4
+ import { Button } from 'primeng/button';
5
+ import { OverlayPanel, OverlayPanelModule } from 'primeng/overlaypanel';
6
+ import { Table } from 'primeng/table';
7
+ import { TagModule } from 'primeng/tag';
8
+ import { COMMONS_MODULE_CONFIG_IT, FilterMatchMode, TemplateDirective, findTemplateByName } from '@mediusinc/mng-commons/core';
9
+ import { filterApplySerializationConfigToCmp, filterGetDateConfig, filterSetMetadataOnChange, filterSetMetadataOnDisplayChange } from '../../../helpers/filters';
10
+ import { DataListService } from '../../../services/data-list.service';
11
+ import { FilterActiveTagComponent } from '../filter-active-tag/filter-active-tag.component';
12
+ import { FilterFormComponent } from '../filter-form/filter-form.component';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "primeng/overlaypanel";
15
+ export class FilterOverlayWithTagComponent {
16
+ constructor() {
17
+ this.primeTable = inject(Table);
18
+ this.dataListService = inject((DataListService));
19
+ this.moduleConfig = inject(COMMONS_MODULE_CONFIG_IT, { optional: true });
20
+ this.descriptors = input.required();
21
+ this.model = input.required();
22
+ this.filterOverlay = viewChild.required(OverlayPanel);
23
+ this.templates = contentChildren(TemplateDirective);
24
+ this.titleTemplate = computed(() => findTemplateByName([...this.templates()], 'title'));
25
+ this.filters = computed(() => {
26
+ return this.setInitialFilters(this.dataListService.filterMeta());
27
+ });
28
+ this.activeFilters = computed(() => {
29
+ const filterEntry = this.filters();
30
+ return this.descriptors()
31
+ .filter(d => {
32
+ const filter = filterEntry[d.property];
33
+ return (filter &&
34
+ filter.matchMode !== undefined &&
35
+ filter.displayMatchMode !== undefined &&
36
+ (filter.matchMode === FilterMatchMode.Exists ||
37
+ filter.matchMode === FilterMatchMode.DoesNotExist ||
38
+ (filter.value !== undefined && filter.displayValue !== undefined)));
39
+ })
40
+ .map(d => filterEntry[d.property]);
41
+ });
42
+ this.activeFilterProperties = computed(() => this.activeFilters().map(f => f.descriptor.property));
43
+ this.filterEdit = signal(undefined);
44
+ this.serializationCfg = {};
45
+ // UI utilities
46
+ this.clickedOnFilterEvent = undefined;
47
+ filterApplySerializationConfigToCmp(this.moduleConfig, this.serializationCfg);
48
+ }
49
+ onFilterFormApply(event) {
50
+ const descriptor = this.descriptors().find(d => d.property === event.property);
51
+ if (!descriptor) {
52
+ return;
53
+ }
54
+ this.updatePrimeTableFilter({
55
+ ...filterSetMetadataOnDisplayChange({ caseSensitive: event.caseSensitive }, event.matchMode, event.value, descriptor.filterType, filterGetDateConfig(descriptor, this.serializationCfg)),
56
+ descriptor: descriptor
57
+ });
58
+ // Resetting the form is done in the onOverlayHide callback because resetting the
59
+ // values here renders empty inputs for a split second before closing the overlay
60
+ this.filterOverlay().hide();
61
+ }
62
+ onFilterRemove(event) {
63
+ this.updatePrimeTableFilter({
64
+ ...this.filters()[event.filter.descriptor.property],
65
+ matchMode: undefined,
66
+ displayMatchMode: undefined,
67
+ value: undefined,
68
+ displayValue: undefined
69
+ });
70
+ }
71
+ onFilterEdit(event) {
72
+ this.onFilterAddOrEdit(event.filter, event.event);
73
+ }
74
+ onFilterAdd(event) {
75
+ this.onFilterAddOrEdit(undefined, event);
76
+ }
77
+ onOverlayHide() {
78
+ if (this.clickedOnFilterEvent) {
79
+ // show the overlay again if the user clicked another filter
80
+ setTimeout(() => {
81
+ this.filterOverlay().show(this.clickedOnFilterEvent);
82
+ this.clickedOnFilterEvent = undefined;
83
+ }, 0);
84
+ return;
85
+ }
86
+ else {
87
+ // clean state
88
+ this.filterEdit.set(undefined);
89
+ }
90
+ }
91
+ onFilterAddOrEdit(filter, event) {
92
+ // If user clicks on the same filter or add that is already open, close the overlay
93
+ if (this.filterOverlay().overlayVisible && this.filterEdit()?.descriptor.property === filter?.descriptor.property) {
94
+ this.filterOverlay().hide();
95
+ this.filterEdit.set(undefined);
96
+ return;
97
+ }
98
+ this.filterEdit.set(filter);
99
+ if (this.filterOverlay().overlayVisible) {
100
+ // In case the user clicks another filter, 're-showing' the overlay in
101
+ // the new position is done in the onOverlayHide callback
102
+ this.clickedOnFilterEvent = event;
103
+ this.filterOverlay().hide();
104
+ }
105
+ else {
106
+ this.filterOverlay().show(event);
107
+ }
108
+ }
109
+ setInitialFilters(filtersInit) {
110
+ const filtersResult = {};
111
+ for (const property in filtersInit) {
112
+ const filterInit = filtersInit[property];
113
+ const descriptor = this.descriptors().find(filterDescriptor => filterDescriptor.property === property);
114
+ if (!descriptor) {
115
+ continue;
116
+ }
117
+ filtersResult[property] = {
118
+ ...filterSetMetadataOnChange(filterInit, filterInit.matchMode, filterInit.value, descriptor.filterType, filterGetDateConfig(descriptor, this.serializationCfg)),
119
+ descriptor: descriptor
120
+ };
121
+ }
122
+ return filtersResult;
123
+ }
124
+ updatePrimeTableFilter(filter) {
125
+ this.primeTable.filters[filter.descriptor.property] = { ...filter };
126
+ this.primeTable._filter();
127
+ }
128
+ 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 }); }
130
+ }
131
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: FilterOverlayWithTagComponent, decorators: [{
132
+ 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" }]
134
+ }], 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"]}