@gnggln/ng-ui-system 1.0.0-alpha.0
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/esm2022/gnggln-ng-ui-system.mjs +5 -0
- package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
- package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
- package/esm2022/lib/components/accordion/index.mjs +2 -0
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
- package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
- package/esm2022/lib/components/base-layout/index.mjs +14 -0
- package/esm2022/lib/components/button/button-area.component.mjs +196 -0
- package/esm2022/lib/components/button/button.component.mjs +164 -0
- package/esm2022/lib/components/button/button.types.mjs +6 -0
- package/esm2022/lib/components/button/index.mjs +16 -0
- package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
- package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
- package/esm2022/lib/components/crud-table/index.mjs +16 -0
- package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
- package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
- package/esm2022/lib/components/form-builder/index.mjs +19 -0
- package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
- package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
- package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
- package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
- package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
- package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
- package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
- package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
- package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
- package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
- package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
- package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
- package/esm2022/lib/components/layout-builder/index.mjs +18 -0
- package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
- package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
- package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
- package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
- package/esm2022/lib/components/modal/index.mjs +4 -0
- package/esm2022/lib/components/modal/modal.component.mjs +139 -0
- package/esm2022/lib/components/modal/modal.service.mjs +194 -0
- package/esm2022/lib/components/modal/modal.types.mjs +6 -0
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
- package/esm2022/lib/components/page-header/index.mjs +20 -0
- package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
- package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
- package/esm2022/lib/components/table/index.mjs +2 -0
- package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
- package/esm2022/lib/components/table/table.types.mjs +6 -0
- package/esm2022/lib/core/types/index.mjs +6 -0
- package/esm2022/lib/core/utils/index.mjs +53 -0
- package/esm2022/lib/sources/location-data.opt.json +8942 -0
- package/esm2022/lib/sources/nazioni.opt.json +215 -0
- package/esm2022/public-api.mjs +34 -0
- package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/accordion/accordion.component.d.ts +118 -0
- package/lib/components/accordion/accordion.types.d.ts +62 -0
- package/lib/components/accordion/index.d.ts +2 -0
- package/lib/components/base-layout/base-layout.component.d.ts +83 -0
- package/lib/components/base-layout/base-layout.types.d.ts +26 -0
- package/lib/components/base-layout/index.d.ts +13 -0
- package/lib/components/button/button-area.component.d.ts +88 -0
- package/lib/components/button/button.component.d.ts +55 -0
- package/lib/components/button/button.types.d.ts +70 -0
- package/lib/components/button/index.d.ts +15 -0
- package/lib/components/crud-table/crud-table.component.d.ts +143 -0
- package/lib/components/crud-table/crud-table.types.d.ts +207 -0
- package/lib/components/crud-table/index.d.ts +15 -0
- package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
- package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
- package/lib/components/form-builder/form-builder.component.d.ts +183 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
- package/lib/components/form-builder/index.d.ts +13 -0
- package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
- package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
- package/lib/components/form-builder/services/location.service.d.ts +83 -0
- package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
- package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
- package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
- package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
- package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
- package/lib/components/form-builder/types/condition.types.d.ts +51 -0
- package/lib/components/form-builder/types/field.types.d.ts +288 -0
- package/lib/components/form-builder/types/index.d.ts +5 -0
- package/lib/components/form-builder/types/schema.types.d.ts +227 -0
- package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
- package/lib/components/form-builder/types/validation.types.d.ts +174 -0
- package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
- package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
- package/lib/components/form-builder-editor/index.d.ts +15 -0
- package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
- package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
- package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
- package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
- package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
- package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
- package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
- package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
- package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
- package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
- package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
- package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
- package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
- package/lib/components/layout-builder/index.d.ts +16 -0
- package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
- package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
- package/lib/components/layout-builder/layout.service.d.ts +100 -0
- package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
- package/lib/components/modal/index.d.ts +4 -0
- package/lib/components/modal/modal.component.d.ts +44 -0
- package/lib/components/modal/modal.service.d.ts +93 -0
- package/lib/components/modal/modal.types.d.ts +110 -0
- package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
- package/lib/components/page-header/index.d.ts +16 -0
- package/lib/components/page-header/page-header.component.d.ts +59 -0
- package/lib/components/page-header/page-header.types.d.ts +96 -0
- package/lib/components/table/index.d.ts +2 -0
- package/lib/components/table/paginated-table.component.d.ts +85 -0
- package/lib/components/table/table.types.d.ts +81 -0
- package/lib/core/types/index.d.ts +57 -0
- package/lib/core/utils/index.d.ts +29 -0
- package/package.json +44 -0
- package/public-api.d.ts +22 -0
|
@@ -0,0 +1,824 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectorRef, ViewEncapsulation, inject, ElementRef, } from '@angular/core';
|
|
2
|
+
import { FormGroup, FormControl, ReactiveFormsModule, } from '@angular/forms';
|
|
3
|
+
import { NgTemplateOutlet } from '@angular/common';
|
|
4
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
5
|
+
import { MatInputModule } from '@angular/material/input';
|
|
6
|
+
import { MatSelectModule } from '@angular/material/select';
|
|
7
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
8
|
+
import { MatSlideToggleModule } from '@angular/material/slide-toggle';
|
|
9
|
+
import { MatRadioModule } from '@angular/material/radio';
|
|
10
|
+
import { MatDatepickerModule } from '@angular/material/datepicker';
|
|
11
|
+
import { MatNativeDateModule, ErrorStateMatcher, DateAdapter, MAT_DATE_FORMATS, MAT_DATE_LOCALE, } from '@angular/material/core';
|
|
12
|
+
import { MatAutocompleteModule } from '@angular/material/autocomplete';
|
|
13
|
+
import { MatChipsModule } from '@angular/material/chips';
|
|
14
|
+
import { MatExpansionModule } from '@angular/material/expansion';
|
|
15
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
16
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
17
|
+
import { Subject } from 'rxjs';
|
|
18
|
+
import { debounceTime, takeUntil } from 'rxjs/operators';
|
|
19
|
+
import { UiButtonAreaComponent } from '../../components/button/button-area.component';
|
|
20
|
+
import { UiFormConditionService } from './services/form-condition.service';
|
|
21
|
+
import { UiFormValidationService } from './services/form-validation.service';
|
|
22
|
+
import { UiFormErrorSummaryComponent } from './sub-components/error-summary/form-error-summary.component';
|
|
23
|
+
import { UiFileInputComponent } from './sub-components/file-input/file-input.component';
|
|
24
|
+
import { UiSpecificaTerritorialeComponent } from './sub-components/specifica-territoriale/specifica-territoriale.component';
|
|
25
|
+
import { UiTableTerritorialeComponent } from './sub-components/table-territoriale/table-territoriale.component';
|
|
26
|
+
import { UiCurrencyInputDirective } from './directives/currency-input.directive';
|
|
27
|
+
import { UiItalianDateAdapter, UI_IT_DATE_FORMATS } from './adapters/it-date-adapter';
|
|
28
|
+
import * as i0 from "@angular/core";
|
|
29
|
+
import * as i1 from "@angular/forms";
|
|
30
|
+
import * as i2 from "@angular/material/form-field";
|
|
31
|
+
import * as i3 from "@angular/material/input";
|
|
32
|
+
import * as i4 from "@angular/material/select";
|
|
33
|
+
import * as i5 from "@angular/material/core";
|
|
34
|
+
import * as i6 from "@angular/material/checkbox";
|
|
35
|
+
import * as i7 from "@angular/material/slide-toggle";
|
|
36
|
+
import * as i8 from "@angular/material/radio";
|
|
37
|
+
import * as i9 from "@angular/material/datepicker";
|
|
38
|
+
import * as i10 from "@angular/material/autocomplete";
|
|
39
|
+
import * as i11 from "@angular/material/chips";
|
|
40
|
+
import * as i12 from "@angular/material/expansion";
|
|
41
|
+
import * as i13 from "@angular/material/tooltip";
|
|
42
|
+
import * as i14 from "lucide-angular";
|
|
43
|
+
/**
|
|
44
|
+
* ErrorStateMatcher custom per il form builder.
|
|
45
|
+
* Mostra lo stato di errore quando il campo e invalido E (touched OPPURE dirty).
|
|
46
|
+
* Sostituisce il matcher di default di Angular Material.
|
|
47
|
+
*/
|
|
48
|
+
export class UiFormErrorStateMatcher {
|
|
49
|
+
isErrorState(control, form) {
|
|
50
|
+
return !!(control && control.invalid && (control.touched || control.dirty));
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Form builder data-driven e schema-based.
|
|
55
|
+
*
|
|
56
|
+
* Genera un form completo a partire da uno schema dichiarativo (`UiFormSchema`).
|
|
57
|
+
* Supporta 16 tipi di campo, validazione condizionale, cross-field validation,
|
|
58
|
+
* date con offset, opzioni dipendenti/dinamiche, sezioni collapsibili,
|
|
59
|
+
* error summary con scroll-to-field e button override.
|
|
60
|
+
*
|
|
61
|
+
* @selector ui-form-builder
|
|
62
|
+
*
|
|
63
|
+
* @example
|
|
64
|
+
* ```html
|
|
65
|
+
* <ui-form-builder
|
|
66
|
+
* [schema]="formSchema"
|
|
67
|
+
* [initialData]="userData"
|
|
68
|
+
* (formSubmit)="onSubmit($event)"
|
|
69
|
+
* (valueChange)="onChange($event)"
|
|
70
|
+
* />
|
|
71
|
+
* ```
|
|
72
|
+
*/
|
|
73
|
+
export class UiFormBuilderComponent {
|
|
74
|
+
constructor() {
|
|
75
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
76
|
+
this.elRef = inject(ElementRef);
|
|
77
|
+
this.conditionService = inject(UiFormConditionService);
|
|
78
|
+
this.validationService = inject(UiFormValidationService);
|
|
79
|
+
this.destroy$ = new Subject();
|
|
80
|
+
/** ID schema corrente -- usato per evitare rebuild superflui. */
|
|
81
|
+
this.currentSchemaId = null;
|
|
82
|
+
/** Dati iniziali per la prevalorizzazione. */
|
|
83
|
+
this.initialData = {};
|
|
84
|
+
/** Modalita sola lettura globale. */
|
|
85
|
+
this.readonly = false;
|
|
86
|
+
/** Stato disabilitato globale. */
|
|
87
|
+
this.disabled = false;
|
|
88
|
+
/** Chiave del pulsante in stato di caricamento. */
|
|
89
|
+
this.loadingFor = null;
|
|
90
|
+
// ─── Output ─────────────────────────────────────────────────────
|
|
91
|
+
this.valueChange = new EventEmitter();
|
|
92
|
+
this.validationChange = new EventEmitter();
|
|
93
|
+
this.formSubmit = new EventEmitter();
|
|
94
|
+
this.formReset = new EventEmitter();
|
|
95
|
+
this.customEvent = new EventEmitter();
|
|
96
|
+
/** Mappa campo -> opzioni filtrate (per autocomplete e dependent). */
|
|
97
|
+
this.filteredOptions = {};
|
|
98
|
+
/** Campi prevalorizzati (mostrano errori subito). */
|
|
99
|
+
this.prevalorizedFields = new Set();
|
|
100
|
+
/** Mappa campo -> regole di validazione (per lookup rapido). */
|
|
101
|
+
this.validationRulesMap = {};
|
|
102
|
+
/** Mappa dipendenze: campoTarget -> campi dipendenti. */
|
|
103
|
+
this.fieldDependencies = new Map();
|
|
104
|
+
/** Flag per evitare emissioni durante l'inizializzazione. */
|
|
105
|
+
this.isInitializing = true;
|
|
106
|
+
}
|
|
107
|
+
// ─── Getter calcolati ───────────────────────────────────────────
|
|
108
|
+
get allFields() {
|
|
109
|
+
return this.schema?.sections?.flatMap((s) => s.fields) || [];
|
|
110
|
+
}
|
|
111
|
+
get formErrors() {
|
|
112
|
+
return this.getDetailedFormErrors();
|
|
113
|
+
}
|
|
114
|
+
get formButtons() {
|
|
115
|
+
if (this.buttonsOverride)
|
|
116
|
+
return this.buttonsOverride;
|
|
117
|
+
if (!this.schema?.config?.showButtons && this.schema?.config?.showButtons !== undefined)
|
|
118
|
+
return [];
|
|
119
|
+
return [
|
|
120
|
+
{ label: this.schema?.config?.buttonLabels?.reset || 'Reset', variant: 'outline', action: () => this.onReset() },
|
|
121
|
+
{
|
|
122
|
+
label: this.schema?.config?.buttonLabels?.submit || 'Invia',
|
|
123
|
+
variant: 'primary',
|
|
124
|
+
action: () => this.onSubmit(),
|
|
125
|
+
},
|
|
126
|
+
];
|
|
127
|
+
}
|
|
128
|
+
// ─── Lifecycle ──────────────────────────────────────────────────
|
|
129
|
+
//
|
|
130
|
+
// ORDINE LIFECYCLE ANGULAR:
|
|
131
|
+
// 1. constructor
|
|
132
|
+
// 2. ngOnChanges (prima chiamata con valori iniziali)
|
|
133
|
+
// 3. ngOnInit
|
|
134
|
+
// 4. ngAfterViewInit
|
|
135
|
+
// 5. ngOnChanges (chiamate successive per cambio input)
|
|
136
|
+
//
|
|
137
|
+
// PER IL WIZARD:
|
|
138
|
+
// Il wizard crea un nuovo schema (con id diverso) ad ogni cambio step.
|
|
139
|
+
// Lo schema viene cachato nel wizard per evitare rebuild infiniti.
|
|
140
|
+
// ngOnChanges rileva il cambio di schema.id e ricostruisce il form.
|
|
141
|
+
ngOnInit() {
|
|
142
|
+
// [DEBUG] Verifica che lo schema sia disponibile al momento di ngOnInit
|
|
143
|
+
console.debug('[UiFormBuilder] ngOnInit - schema:', this.schema?.id, '| formGroup definito:', !!this.formGroup);
|
|
144
|
+
if (!this.schema) {
|
|
145
|
+
console.warn('[UiFormBuilder] ngOnInit - schema non presente, il form non verra costruito ora');
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
this.initFormFromSchema();
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Rileva cambiamenti sugli @Input.
|
|
152
|
+
* Ricostruisce il form quando cambia lo schema (id diverso).
|
|
153
|
+
* Cruciale per il wizard che cambia schema ad ogni step.
|
|
154
|
+
*/
|
|
155
|
+
ngOnChanges(changes) {
|
|
156
|
+
// [DEBUG] Log di tutti i cambiamenti degli Input
|
|
157
|
+
const changedInputs = Object.keys(changes)
|
|
158
|
+
.map((k) => `${k}(firstChange=${changes[k].firstChange})`)
|
|
159
|
+
.join(', ');
|
|
160
|
+
console.debug('[UiFormBuilder] ngOnChanges -', changedInputs);
|
|
161
|
+
if (changes['schema'] && !changes['schema'].firstChange) {
|
|
162
|
+
const newSchema = changes['schema'].currentValue;
|
|
163
|
+
console.debug('[UiFormBuilder] ngOnChanges - nuovo schema:', newSchema?.id, '| corrente:', this.currentSchemaId);
|
|
164
|
+
if (newSchema && newSchema.id !== this.currentSchemaId) {
|
|
165
|
+
console.debug('[UiFormBuilder] ngOnChanges - REBUILD: id diverso, ricostruisco il form');
|
|
166
|
+
this.destroyForm();
|
|
167
|
+
this.initFormFromSchema();
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
// Aggiorna lo stato disabled/readonly globalmente
|
|
171
|
+
if (changes['disabled'] && !changes['disabled'].firstChange && this.formGroup) {
|
|
172
|
+
this.allFields.forEach((field) => {
|
|
173
|
+
const control = this.formGroup.get(field.key);
|
|
174
|
+
if (!control)
|
|
175
|
+
return;
|
|
176
|
+
if (this.disabled && control.enabled) {
|
|
177
|
+
control.disable({ emitEvent: false });
|
|
178
|
+
}
|
|
179
|
+
else if (!this.disabled && control.disabled && !field.disabled) {
|
|
180
|
+
control.enable({ emitEvent: false });
|
|
181
|
+
}
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
ngAfterViewInit() {
|
|
186
|
+
// [DEBUG] Verifica stato dopo la prima renderizzazione della vista
|
|
187
|
+
console.debug('[UiFormBuilder] ngAfterViewInit - formGroup definito:', !!this.formGroup, '| campi:', this.formGroup ? Object.keys(this.formGroup.controls).length : 0);
|
|
188
|
+
}
|
|
189
|
+
ngOnDestroy() {
|
|
190
|
+
console.debug('[UiFormBuilder] ngOnDestroy - schema:', this.currentSchemaId);
|
|
191
|
+
this.destroy$.next();
|
|
192
|
+
this.destroy$.complete();
|
|
193
|
+
this.valueChangeSub?.unsubscribe();
|
|
194
|
+
}
|
|
195
|
+
/**
|
|
196
|
+
* Inizializza il form dallo schema corrente.
|
|
197
|
+
* Estratto per poter essere richiamato sia da ngOnInit che da ngOnChanges.
|
|
198
|
+
*/
|
|
199
|
+
initFormFromSchema() {
|
|
200
|
+
if (!this.schema) {
|
|
201
|
+
console.warn('[UiFormBuilder] initFormFromSchema - schema nullo, skip');
|
|
202
|
+
return;
|
|
203
|
+
}
|
|
204
|
+
console.debug('[UiFormBuilder] initFormFromSchema - costruisco form per:', this.schema.id, '| sezioni:', this.schema.sections?.length, '| campi totali:', this.schema.sections?.reduce((sum, s) => sum + s.fields.length, 0));
|
|
205
|
+
this.currentSchemaId = this.schema.id;
|
|
206
|
+
this.buildForm();
|
|
207
|
+
this.isInitializing = false;
|
|
208
|
+
console.debug('[UiFormBuilder] initFormFromSchema - form costruito. Controls:', Object.keys(this.formGroup.controls));
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* Pulisce lo stato del form prima di un rebuild.
|
|
212
|
+
*/
|
|
213
|
+
destroyForm() {
|
|
214
|
+
console.debug('[UiFormBuilder] destroyForm - pulizia stato per schema:', this.currentSchemaId);
|
|
215
|
+
this.valueChangeSub?.unsubscribe();
|
|
216
|
+
this.valueChangeSub = undefined;
|
|
217
|
+
this.filteredOptions = {};
|
|
218
|
+
this.prevalorizedFields.clear();
|
|
219
|
+
this.validationRulesMap = {};
|
|
220
|
+
this.fieldDependencies.clear();
|
|
221
|
+
this.isInitializing = true;
|
|
222
|
+
this.currentSchemaId = null;
|
|
223
|
+
}
|
|
224
|
+
// ─── Costruzione form ───────────────────────────────────────────
|
|
225
|
+
buildForm() {
|
|
226
|
+
const controls = {};
|
|
227
|
+
for (const field of this.allFields) {
|
|
228
|
+
const initialValue = this.initialData?.[field.key] ?? field.defaultValue ?? null;
|
|
229
|
+
const isDisabled = field.disabled || this.disabled;
|
|
230
|
+
// Crea il FormControl
|
|
231
|
+
const control = new FormControl({ value: initialValue, disabled: isDisabled });
|
|
232
|
+
// Registra le regole di validazione
|
|
233
|
+
if (field.validation?.length) {
|
|
234
|
+
this.validationRulesMap[field.key] = field.validation;
|
|
235
|
+
}
|
|
236
|
+
// Traccia campi prevalorizzati
|
|
237
|
+
if (initialValue != null && initialValue !== '' && initialValue !== false) {
|
|
238
|
+
this.prevalorizedFields.add(field.key);
|
|
239
|
+
}
|
|
240
|
+
// Inizializza opzioni filtrate per select/multiselect
|
|
241
|
+
if (field.options && !this.isObservable(field.options)) {
|
|
242
|
+
this.filteredOptions[field.key] = [...field.options];
|
|
243
|
+
}
|
|
244
|
+
controls[field.key] = control;
|
|
245
|
+
}
|
|
246
|
+
this.formGroup = new FormGroup(controls);
|
|
247
|
+
// Applica validatori iniziali
|
|
248
|
+
this.applyAllValidators();
|
|
249
|
+
// Registra dipendenze
|
|
250
|
+
this.buildDependencyMap();
|
|
251
|
+
// Sottoscrivi ai cambiamenti di valore
|
|
252
|
+
this.valueChangeSub = this.formGroup.valueChanges.pipe(debounceTime(50), takeUntil(this.destroy$)).subscribe(() => {
|
|
253
|
+
this.evaluateAllConditions();
|
|
254
|
+
this.updateDependentOptions();
|
|
255
|
+
if (!this.isInitializing) {
|
|
256
|
+
this.valueChange.emit(this.getFormValue());
|
|
257
|
+
this.validationChange.emit(this.getValidationState());
|
|
258
|
+
this.cdr.markForCheck();
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
// Valutazione condizioni iniziale
|
|
262
|
+
this.evaluateAllConditions();
|
|
263
|
+
}
|
|
264
|
+
// ─── Validazione ────────────────────────────────────────────────
|
|
265
|
+
applyAllValidators() {
|
|
266
|
+
for (const field of this.allFields) {
|
|
267
|
+
if (field.validation?.length) {
|
|
268
|
+
const formData = this.formGroup.getRawValue();
|
|
269
|
+
this.validationService.updateFieldValidators(this.formGroup.get(field.key), field.validation, formData);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
// ─── Condizioni ─────────────────────────────────────────────────
|
|
274
|
+
evaluateAllConditions() {
|
|
275
|
+
const formData = this.formGroup.getRawValue();
|
|
276
|
+
for (const field of this.allFields) {
|
|
277
|
+
// Visibilita
|
|
278
|
+
if (field.conditions?.length) {
|
|
279
|
+
const visible = this.conditionService.evaluateConditions(field.conditions, formData);
|
|
280
|
+
const control = this.formGroup.get(field.key);
|
|
281
|
+
if (control) {
|
|
282
|
+
if (!visible && control.enabled) {
|
|
283
|
+
control.disable({ emitEvent: false });
|
|
284
|
+
if (!field.freezeValueOnDisable) {
|
|
285
|
+
control.setValue(field.defaultValue ?? null, { emitEvent: false });
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
else if (visible && control.disabled && !field.disabled && !this.disabled) {
|
|
289
|
+
control.enable({ emitEvent: false });
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
// Disabilita condizionale
|
|
294
|
+
if (field.disableConditions?.length && !field.conditions?.length) {
|
|
295
|
+
const shouldDisable = this.conditionService.evaluateConditions(field.disableConditions, formData);
|
|
296
|
+
const control = this.formGroup.get(field.key);
|
|
297
|
+
if (control) {
|
|
298
|
+
if (shouldDisable && control.enabled) {
|
|
299
|
+
control.disable({ emitEvent: false });
|
|
300
|
+
if (!field.freezeValueOnDisable) {
|
|
301
|
+
control.setValue(field.defaultValue ?? null, { emitEvent: false });
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
else if (!shouldDisable && control.disabled && !field.disabled && !this.disabled) {
|
|
305
|
+
control.enable({ emitEvent: false });
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
// Aggiorna validatori condizionali
|
|
310
|
+
if (field.validation?.some((r) => r.conditions?.length)) {
|
|
311
|
+
this.validationService.updateFieldValidators(this.formGroup.get(field.key), field.validation, formData);
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
// ─── Dipendenze opzioni ─────────────────────────────────────────
|
|
316
|
+
buildDependencyMap() {
|
|
317
|
+
for (const field of this.allFields) {
|
|
318
|
+
if (field.dependentOptions?.dependsOn) {
|
|
319
|
+
const depKey = field.dependentOptions.dependsOn;
|
|
320
|
+
if (!this.fieldDependencies.has(depKey)) {
|
|
321
|
+
this.fieldDependencies.set(depKey, new Set());
|
|
322
|
+
}
|
|
323
|
+
this.fieldDependencies.get(depKey).add(field.key);
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
updateDependentOptions() {
|
|
328
|
+
const formData = this.formGroup.getRawValue();
|
|
329
|
+
for (const field of this.allFields) {
|
|
330
|
+
// Dynamic options
|
|
331
|
+
if (field.dynamicOptions) {
|
|
332
|
+
if (field.dynamicOptions.type === 'year-range') {
|
|
333
|
+
const min = field.dynamicOptions.minYear || new Date().getFullYear() - 10;
|
|
334
|
+
const max = field.dynamicOptions.maxYear || new Date().getFullYear();
|
|
335
|
+
const opts = [];
|
|
336
|
+
for (let y = max; y >= min; y--) {
|
|
337
|
+
opts.push({ value: y, label: String(y) });
|
|
338
|
+
}
|
|
339
|
+
this.filteredOptions[field.key] = opts;
|
|
340
|
+
}
|
|
341
|
+
else if (field.dynamicOptions.generator) {
|
|
342
|
+
this.filteredOptions[field.key] = field.dynamicOptions.generator(formData);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
// Dependent options
|
|
346
|
+
if (field.dependentOptions && field.options && !this.isObservable(field.options)) {
|
|
347
|
+
const dep = field.dependentOptions;
|
|
348
|
+
const depValue = formData[dep.dependsOn];
|
|
349
|
+
const allOptions = field.options;
|
|
350
|
+
if (depValue == null || depValue === '') {
|
|
351
|
+
this.filteredOptions[field.key] = [...allOptions];
|
|
352
|
+
continue;
|
|
353
|
+
}
|
|
354
|
+
this.filteredOptions[field.key] = allOptions.filter((opt) => {
|
|
355
|
+
if (dep.filterType === 'custom' && dep.filterFn) {
|
|
356
|
+
return dep.filterFn(opt, depValue, formData);
|
|
357
|
+
}
|
|
358
|
+
const optVal = Number(opt.value);
|
|
359
|
+
const depVal = Number(depValue);
|
|
360
|
+
if (isNaN(optVal) || isNaN(depVal))
|
|
361
|
+
return true;
|
|
362
|
+
switch (dep.filterType) {
|
|
363
|
+
case 'greater_than':
|
|
364
|
+
return optVal > depVal;
|
|
365
|
+
case 'greater_equal':
|
|
366
|
+
return optVal >= depVal;
|
|
367
|
+
case 'less_than':
|
|
368
|
+
return optVal < depVal;
|
|
369
|
+
case 'less_equal':
|
|
370
|
+
return optVal <= depVal;
|
|
371
|
+
case 'equals':
|
|
372
|
+
return optVal === depVal;
|
|
373
|
+
case 'not_equals':
|
|
374
|
+
return optVal !== depVal;
|
|
375
|
+
default:
|
|
376
|
+
return true;
|
|
377
|
+
}
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
// ─── API pubblica ───────────────────────────────────────────────
|
|
383
|
+
/** Verifica se un campo e visibile. */
|
|
384
|
+
isFieldVisible(fieldKey) {
|
|
385
|
+
const field = this.allFields.find((f) => f.key === fieldKey);
|
|
386
|
+
if (!field?.conditions?.length)
|
|
387
|
+
return true;
|
|
388
|
+
return this.conditionService.evaluateConditions(field.conditions, this.formGroup.getRawValue());
|
|
389
|
+
}
|
|
390
|
+
/** Verifica se un campo e obbligatorio (include validazione condizionale). */
|
|
391
|
+
isFieldRequired(fieldKey) {
|
|
392
|
+
const field = this.allFields.find((f) => f.key === fieldKey);
|
|
393
|
+
if (field?.required)
|
|
394
|
+
return true;
|
|
395
|
+
const rules = field?.validation || [];
|
|
396
|
+
const formData = this.formGroup.getRawValue();
|
|
397
|
+
return rules.some((r) => {
|
|
398
|
+
if (r.type !== 'required')
|
|
399
|
+
return false;
|
|
400
|
+
if (!r.conditions?.length)
|
|
401
|
+
return true;
|
|
402
|
+
return this.conditionService.evaluateConditions(r.conditions, formData);
|
|
403
|
+
});
|
|
404
|
+
}
|
|
405
|
+
/** Verifica se una sezione e visibile. */
|
|
406
|
+
isSectionVisible(sectionId) {
|
|
407
|
+
const section = this.schema.sections.find((s) => s.id === sectionId);
|
|
408
|
+
if (!section?.conditions?.length)
|
|
409
|
+
return true;
|
|
410
|
+
return this.conditionService.evaluateConditions(section.conditions, this.formGroup.getRawValue());
|
|
411
|
+
}
|
|
412
|
+
/** Verifica se mostrare errori per un campo. */
|
|
413
|
+
shouldShowFieldErrors(fieldKey) {
|
|
414
|
+
const control = this.formGroup.get(fieldKey);
|
|
415
|
+
if (!control)
|
|
416
|
+
return false;
|
|
417
|
+
return control.invalid && (control.dirty || control.touched || this.prevalorizedFields.has(fieldKey));
|
|
418
|
+
}
|
|
419
|
+
/** Errori di un campo come array di messaggi. */
|
|
420
|
+
getFieldErrors(fieldKey) {
|
|
421
|
+
const control = this.formGroup.get(fieldKey);
|
|
422
|
+
if (!control?.errors)
|
|
423
|
+
return [];
|
|
424
|
+
const field = this.allFields.find((f) => f.key === fieldKey);
|
|
425
|
+
const rules = field?.validation || [];
|
|
426
|
+
const messages = [];
|
|
427
|
+
for (const [errorKey] of Object.entries(control.errors)) {
|
|
428
|
+
// Angular usa chiavi lowercase (es. "minlength"), lo schema usa camelCase (es. "minLength").
|
|
429
|
+
// Normalizziamo per trovare la corrispondenza corretta.
|
|
430
|
+
const rule = rules.find((r) => this.normalizeValidationKey(r.type) === errorKey || (r.type === 'crossField' && errorKey === 'crossField'));
|
|
431
|
+
if (rule?.message) {
|
|
432
|
+
messages.push(rule.message);
|
|
433
|
+
}
|
|
434
|
+
else {
|
|
435
|
+
messages.push(this.getDefaultErrorMessage(errorKey, control.errors[errorKey]));
|
|
436
|
+
}
|
|
437
|
+
}
|
|
438
|
+
return messages;
|
|
439
|
+
}
|
|
440
|
+
/** Valore completo del form (include campi disabilitati). */
|
|
441
|
+
getFormValue() {
|
|
442
|
+
return this.formGroup.getRawValue();
|
|
443
|
+
}
|
|
444
|
+
/** Il form e valido. */
|
|
445
|
+
isFormValid() {
|
|
446
|
+
return this.formGroup.valid;
|
|
447
|
+
}
|
|
448
|
+
/** Stato di validazione completo. */
|
|
449
|
+
getValidationState() {
|
|
450
|
+
const errors = {};
|
|
451
|
+
const dirty = [];
|
|
452
|
+
const touched = [];
|
|
453
|
+
for (const field of this.allFields) {
|
|
454
|
+
const control = this.formGroup.get(field.key);
|
|
455
|
+
if (!control)
|
|
456
|
+
continue;
|
|
457
|
+
if (control.dirty)
|
|
458
|
+
dirty.push(field.key);
|
|
459
|
+
if (control.touched)
|
|
460
|
+
touched.push(field.key);
|
|
461
|
+
const fieldErrors = this.getFieldErrors(field.key);
|
|
462
|
+
if (fieldErrors.length)
|
|
463
|
+
errors[field.key] = fieldErrors;
|
|
464
|
+
}
|
|
465
|
+
return { valid: this.formGroup.valid, errors, dirty, touched };
|
|
466
|
+
}
|
|
467
|
+
/** Errori dettagliati per l'error summary. */
|
|
468
|
+
getDetailedFormErrors() {
|
|
469
|
+
const details = [];
|
|
470
|
+
for (const field of this.allFields) {
|
|
471
|
+
if (field.type === 'flag' || field.type === 'divider')
|
|
472
|
+
continue;
|
|
473
|
+
if (!this.isFieldVisible(field.key))
|
|
474
|
+
continue;
|
|
475
|
+
const control = this.formGroup.get(field.key);
|
|
476
|
+
if (!control?.errors)
|
|
477
|
+
continue;
|
|
478
|
+
const errors = this.getFieldErrors(field.key);
|
|
479
|
+
if (errors.length) {
|
|
480
|
+
details.push({ fieldKey: field.key, fieldLabel: field.label, errors });
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
return details;
|
|
484
|
+
}
|
|
485
|
+
/** Conta errori totali. */
|
|
486
|
+
getFormErrorsCount() {
|
|
487
|
+
return this.getDetailedFormErrors().reduce((sum, d) => sum + d.errors.length, 0);
|
|
488
|
+
}
|
|
489
|
+
/** Controlla se ci sono errori. */
|
|
490
|
+
hasFormErrors() {
|
|
491
|
+
return this.getFormErrorsCount() > 0;
|
|
492
|
+
}
|
|
493
|
+
/** Controllo per un campo. */
|
|
494
|
+
getFieldControl(key) {
|
|
495
|
+
return this.formGroup.get(key) ?? null;
|
|
496
|
+
}
|
|
497
|
+
/** Aggiorna il valore di un campo programmaticamente. */
|
|
498
|
+
updateFieldValue(key, value) {
|
|
499
|
+
const control = this.formGroup.get(key);
|
|
500
|
+
if (control) {
|
|
501
|
+
control.setValue(value);
|
|
502
|
+
control.markAsDirty();
|
|
503
|
+
}
|
|
504
|
+
}
|
|
505
|
+
/** Forza la validazione di tutti i campi. */
|
|
506
|
+
validateAllFields() {
|
|
507
|
+
Object.keys(this.formGroup.controls).forEach((key) => {
|
|
508
|
+
const control = this.formGroup.get(key);
|
|
509
|
+
control?.markAsTouched();
|
|
510
|
+
control?.markAsDirty();
|
|
511
|
+
control?.updateValueAndValidity();
|
|
512
|
+
});
|
|
513
|
+
this.cdr.markForCheck();
|
|
514
|
+
}
|
|
515
|
+
/** Scrolla al campo e lo evidenzia. */
|
|
516
|
+
scrollToField(fieldKey) {
|
|
517
|
+
// Forza lo stato dirty e touched per mostrare immediatamente gli errori
|
|
518
|
+
const control = this.formGroup.get(fieldKey);
|
|
519
|
+
if (control) {
|
|
520
|
+
control.markAsTouched();
|
|
521
|
+
control.markAsDirty();
|
|
522
|
+
control.updateValueAndValidity();
|
|
523
|
+
}
|
|
524
|
+
// Delay per lasciare che Angular Material aggiorni il DOM
|
|
525
|
+
setTimeout(() => {
|
|
526
|
+
// Cerca l'elemento tramite diversi selettori in ordine di priorita
|
|
527
|
+
const selectors = [`[data-field-key="${fieldKey}"]`, `[formControlName="${fieldKey}"]`];
|
|
528
|
+
let targetElement = null;
|
|
529
|
+
for (const sel of selectors) {
|
|
530
|
+
targetElement = this.elRef.nativeElement.querySelector(sel);
|
|
531
|
+
if (targetElement)
|
|
532
|
+
break;
|
|
533
|
+
}
|
|
534
|
+
if (!targetElement)
|
|
535
|
+
return;
|
|
536
|
+
// Se trovato il formControlName, risali al wrapper
|
|
537
|
+
if (targetElement.hasAttribute('formControlName')) {
|
|
538
|
+
const wrapper = targetElement.closest('.ui-form-builder__field-wrapper, mat-form-field');
|
|
539
|
+
if (wrapper)
|
|
540
|
+
targetElement = wrapper;
|
|
541
|
+
}
|
|
542
|
+
targetElement.scrollIntoView({ behavior: 'smooth', block: 'center' });
|
|
543
|
+
// Aggiungi classe highlight e rimuovila dopo l'animazione
|
|
544
|
+
targetElement.classList.add('ui-form-field--highlight');
|
|
545
|
+
setTimeout(() => targetElement.classList.remove('ui-form-field--highlight'), 2000);
|
|
546
|
+
}, 100);
|
|
547
|
+
}
|
|
548
|
+
/** Submit del form. */
|
|
549
|
+
onSubmit() {
|
|
550
|
+
this.validateAllFields();
|
|
551
|
+
if (this.formGroup.valid) {
|
|
552
|
+
this.formSubmit.emit(this.getFormValue());
|
|
553
|
+
}
|
|
554
|
+
}
|
|
555
|
+
/** Reset del form. */
|
|
556
|
+
onReset() {
|
|
557
|
+
// Resetta a valori iniziali o default
|
|
558
|
+
for (const field of this.allFields) {
|
|
559
|
+
const value = this.initialData?.[field.key] ?? field.defaultValue ?? null;
|
|
560
|
+
const control = this.formGroup.get(field.key);
|
|
561
|
+
control?.setValue(value, { emitEvent: false });
|
|
562
|
+
control?.markAsPristine();
|
|
563
|
+
control?.markAsUntouched();
|
|
564
|
+
}
|
|
565
|
+
this.formGroup.updateValueAndValidity();
|
|
566
|
+
this.formReset.emit();
|
|
567
|
+
this.cdr.markForCheck();
|
|
568
|
+
}
|
|
569
|
+
// ─── Template helpers ───────────────────────────────────────────
|
|
570
|
+
/** Opzioni per un campo (filtrando per autocomplete query). */
|
|
571
|
+
getFieldOptions(field) {
|
|
572
|
+
return this.filteredOptions[field.key] || [];
|
|
573
|
+
}
|
|
574
|
+
/** Filtra opzioni per autocomplete. */
|
|
575
|
+
filterOptions(field, query) {
|
|
576
|
+
if (!query) {
|
|
577
|
+
if (field.options && !this.isObservable(field.options)) {
|
|
578
|
+
this.filteredOptions[field.key] = [...field.options];
|
|
579
|
+
}
|
|
580
|
+
return;
|
|
581
|
+
}
|
|
582
|
+
const lowerQuery = query.toLowerCase();
|
|
583
|
+
if (field.asyncOptions) {
|
|
584
|
+
field.asyncOptions(query).then((opts) => {
|
|
585
|
+
this.filteredOptions[field.key] = opts;
|
|
586
|
+
this.cdr.markForCheck();
|
|
587
|
+
});
|
|
588
|
+
}
|
|
589
|
+
else if (field.options && !this.isObservable(field.options)) {
|
|
590
|
+
this.filteredOptions[field.key] = field.options.filter((o) => o.label.toLowerCase().includes(lowerQuery));
|
|
591
|
+
}
|
|
592
|
+
}
|
|
593
|
+
/** Display fn per autocomplete. */
|
|
594
|
+
displayFn(options) {
|
|
595
|
+
return (value) => {
|
|
596
|
+
const opt = options.find((o) => o.value === value);
|
|
597
|
+
return opt?.label || '';
|
|
598
|
+
};
|
|
599
|
+
}
|
|
600
|
+
/** Seleziona tutte le opzioni per multiselect. */
|
|
601
|
+
selectAll(field) {
|
|
602
|
+
const options = this.getFieldOptions(field);
|
|
603
|
+
const allValues = options.filter((o) => !o.disabled).map((o) => o.value);
|
|
604
|
+
this.formGroup.get(field.key)?.setValue(allValues);
|
|
605
|
+
}
|
|
606
|
+
/** Aggiunge un chip al freemultiselect (da token: Enter o virgola). */
|
|
607
|
+
addFreeChip(field, event) {
|
|
608
|
+
const value = (event.value || '').trim();
|
|
609
|
+
if (!value)
|
|
610
|
+
return;
|
|
611
|
+
const current = this.formGroup.get(field.key)?.value || [];
|
|
612
|
+
if (!current.includes(value)) {
|
|
613
|
+
this.formGroup.get(field.key)?.setValue([...current, value]);
|
|
614
|
+
}
|
|
615
|
+
event.chipInput?.clear();
|
|
616
|
+
}
|
|
617
|
+
/** Aggiunge un chip al freemultiselect sull'evento blur dell'input. */
|
|
618
|
+
addFreeChipOnBlur(field, event) {
|
|
619
|
+
const input = event.target;
|
|
620
|
+
const value = (input.value || '').trim();
|
|
621
|
+
if (!value)
|
|
622
|
+
return;
|
|
623
|
+
const current = this.formGroup.get(field.key)?.value || [];
|
|
624
|
+
if (!current.includes(value)) {
|
|
625
|
+
this.formGroup.get(field.key)?.setValue([...current, value]);
|
|
626
|
+
}
|
|
627
|
+
input.value = '';
|
|
628
|
+
}
|
|
629
|
+
/** Rimuove un chip dal freemultiselect. */
|
|
630
|
+
removeFreeChip(field, chipValue) {
|
|
631
|
+
const current = this.formGroup.get(field.key)?.value || [];
|
|
632
|
+
const protected_ = field.metadata?.['cantDeleteList'] || [];
|
|
633
|
+
if (protected_.includes(chipValue))
|
|
634
|
+
return;
|
|
635
|
+
this.formGroup.get(field.key)?.setValue(current.filter((v) => v !== chipValue));
|
|
636
|
+
}
|
|
637
|
+
/** Verifica se un chip e protetto dalla cancellazione. */
|
|
638
|
+
isChipProtected(field, chipValue) {
|
|
639
|
+
return (field.metadata?.['cantDeleteList'] || []).includes(chipValue);
|
|
640
|
+
}
|
|
641
|
+
/** Rimuove chip da multiselect. */
|
|
642
|
+
removeMultiselectChip(field, chipValue) {
|
|
643
|
+
const current = this.formGroup.get(field.key)?.value || [];
|
|
644
|
+
this.formGroup.get(field.key)?.setValue(current.filter((v) => v !== chipValue));
|
|
645
|
+
}
|
|
646
|
+
/** Trova label di un'opzione dal valore. */
|
|
647
|
+
getOptionLabel(field, value) {
|
|
648
|
+
const options = this.getFieldOptions(field);
|
|
649
|
+
return options.find((o) => o.value === value)?.label || String(value);
|
|
650
|
+
}
|
|
651
|
+
/** Tipo di input nativo per il campo. */
|
|
652
|
+
getNativeInputType(field) {
|
|
653
|
+
switch (field.type) {
|
|
654
|
+
case 'password':
|
|
655
|
+
return 'password';
|
|
656
|
+
case 'email':
|
|
657
|
+
return 'email';
|
|
658
|
+
default:
|
|
659
|
+
return 'text';
|
|
660
|
+
}
|
|
661
|
+
}
|
|
662
|
+
/**
|
|
663
|
+
* Genera le classi CSS per il wrapper di un campo,
|
|
664
|
+
* combinando eventuali cssClasses dallo schema con le classi
|
|
665
|
+
* responsive per il sistema a griglia base-12.
|
|
666
|
+
*
|
|
667
|
+
* Mobile-first: tutti i campi sono span-12 di default (100%).
|
|
668
|
+
* Le classi `ui-col-{breakpoint}-{n}` attivano la larghezza
|
|
669
|
+
* configurata solo dal breakpoint indicato in su.
|
|
670
|
+
*
|
|
671
|
+
* Breakpoint supportati: sm (≥576px), md (≥768px), lg (≥1024px), xl (≥1280px).
|
|
672
|
+
*/
|
|
673
|
+
getFieldWrapperClasses(field) {
|
|
674
|
+
const classes = [];
|
|
675
|
+
if (field.cssClasses?.length) {
|
|
676
|
+
classes.push(...field.cssClasses);
|
|
677
|
+
}
|
|
678
|
+
// layout.columns → classe per il breakpoint md (default desktop)
|
|
679
|
+
if (field.layout?.columns) {
|
|
680
|
+
classes.push(`ui-col-md-${field.layout.columns}`);
|
|
681
|
+
}
|
|
682
|
+
// layout.breakpoints → classi per breakpoint specifici
|
|
683
|
+
if (field.layout?.breakpoints) {
|
|
684
|
+
for (const [bp, cols] of Object.entries(field.layout.breakpoints)) {
|
|
685
|
+
classes.push(`ui-col-${bp}-${cols}`);
|
|
686
|
+
}
|
|
687
|
+
}
|
|
688
|
+
return classes.join(' ');
|
|
689
|
+
}
|
|
690
|
+
/** Character count per textarea/text con maxLength. */
|
|
691
|
+
getCharCount(fieldKey) {
|
|
692
|
+
const field = this.allFields.find((f) => f.key === fieldKey);
|
|
693
|
+
const maxRule = field?.validation?.find((r) => r.type === 'maxLength');
|
|
694
|
+
if (!maxRule?.value)
|
|
695
|
+
return null;
|
|
696
|
+
const control = this.formGroup.get(fieldKey);
|
|
697
|
+
return {
|
|
698
|
+
current: (control?.value || '').length,
|
|
699
|
+
max: Number(maxRule.value),
|
|
700
|
+
};
|
|
701
|
+
}
|
|
702
|
+
// ─── Utilita private ────────────────────────────────────────────
|
|
703
|
+
isObservable(value) {
|
|
704
|
+
return value && typeof value.subscribe === 'function';
|
|
705
|
+
}
|
|
706
|
+
/**
|
|
707
|
+
* Normalizza la chiave del tipo di validazione dallo schema (camelCase)
|
|
708
|
+
* alla chiave di errore usata da Angular (lowercase).
|
|
709
|
+
* Es: "minLength" -> "minlength", "maxLength" -> "maxlength".
|
|
710
|
+
*/
|
|
711
|
+
normalizeValidationKey(schemaType) {
|
|
712
|
+
const keyMap = {
|
|
713
|
+
minLength: 'minlength',
|
|
714
|
+
maxLength: 'maxlength',
|
|
715
|
+
};
|
|
716
|
+
return keyMap[schemaType] || schemaType;
|
|
717
|
+
}
|
|
718
|
+
getDefaultErrorMessage(errorKey, errorValue) {
|
|
719
|
+
// Se il valore dell'errore e gia un messaggio descrittivo (stringa),
|
|
720
|
+
// lo restituisce direttamente. Questo supporta i validatori personalizzati
|
|
721
|
+
// (es. location, location-table) che forniscono messaggi espliciti come
|
|
722
|
+
// valore dell'errore (es. { nazioneRequired: 'Seleziona una nazione' }).
|
|
723
|
+
if (typeof errorValue === 'string') {
|
|
724
|
+
return errorValue;
|
|
725
|
+
}
|
|
726
|
+
switch (errorKey) {
|
|
727
|
+
case 'required':
|
|
728
|
+
return 'Campo obbligatorio';
|
|
729
|
+
case 'email':
|
|
730
|
+
return 'Formato email non valido';
|
|
731
|
+
case 'min':
|
|
732
|
+
return `Valore minimo: ${errorValue.min}`;
|
|
733
|
+
case 'max':
|
|
734
|
+
return `Valore massimo: ${errorValue.max}`;
|
|
735
|
+
case 'minlength':
|
|
736
|
+
return `Lunghezza minima: ${errorValue.requiredLength} caratteri`;
|
|
737
|
+
case 'maxlength':
|
|
738
|
+
return `Lunghezza massima: ${errorValue.requiredLength} caratteri`;
|
|
739
|
+
case 'pattern':
|
|
740
|
+
return 'Formato non valido';
|
|
741
|
+
case 'crossField':
|
|
742
|
+
return 'Valore non coerente con il campo correlato';
|
|
743
|
+
case 'date-min':
|
|
744
|
+
return errorValue?.message || 'Data troppo remota';
|
|
745
|
+
case 'date-max':
|
|
746
|
+
return errorValue?.message || 'Data troppo avanzata';
|
|
747
|
+
case 'fileSize':
|
|
748
|
+
return `Dimensione massima: ${errorValue?.maxSizeFormatted}`;
|
|
749
|
+
case 'fileType':
|
|
750
|
+
return 'Formato file non accettato';
|
|
751
|
+
case 'fileCount':
|
|
752
|
+
return `Massimo ${errorValue?.maxCount} file`;
|
|
753
|
+
default:
|
|
754
|
+
return 'Valore non valido';
|
|
755
|
+
}
|
|
756
|
+
}
|
|
757
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormBuilderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
758
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiFormBuilderComponent, isStandalone: true, selector: "ui-form-builder", inputs: { schema: "schema", initialData: "initialData", readonly: "readonly", disabled: "disabled", buttonsOverride: "buttonsOverride", loadingFor: "loadingFor" }, outputs: { valueChange: "valueChange", validationChange: "validationChange", formSubmit: "formSubmit", formReset: "formReset", customEvent: "customEvent" }, host: { classAttribute: "ui-form-builder-host" }, providers: [
|
|
759
|
+
{ provide: ErrorStateMatcher, useClass: UiFormErrorStateMatcher },
|
|
760
|
+
// Locale italiano per il datepicker Material
|
|
761
|
+
{ provide: MAT_DATE_LOCALE, useValue: 'it-IT' },
|
|
762
|
+
// Adapter personalizzato per parsing DD/MM/YYYY
|
|
763
|
+
{ provide: DateAdapter, useClass: UiItalianDateAdapter },
|
|
764
|
+
// Formati data italiani (DD/MM/YYYY)
|
|
765
|
+
{ provide: MAT_DATE_FORMATS, useValue: UI_IT_DATE_FORMATS },
|
|
766
|
+
], usesOnChanges: true, ngImport: i0, template: "<!--\r\n ============================================================\r\n UI FORM BUILDER - TEMPLATE PRINCIPALE\r\n ============================================================\r\n ARCHITETTURA:\r\n Ogni <ng-template> che contiene direttive [formControlName]\r\n DEVE wrappare il proprio contenuto in un <div [formGroup]>.\r\n Questo perche Angular risolve l'injector di formControlName\r\n dal CONTESTO DI DICHIARAZIONE del template, non dal punto\r\n di inserimento (ngTemplateOutlet).\r\n\r\n Senza questo wrapper, formControlName non trova il\r\n FormGroupDirective e lancia NG01050.\r\n\r\n Ref: versione tailored (ang-ms-webapp) usa lo stesso pattern.\r\n ============================================================\r\n-->\r\n\r\n@if (schema && formGroup) {\r\n <div class=\"ui-form-builder\" [formGroup]=\"formGroup\" [class]=\"schema.config?.cssClasses?.join(' ')\">\r\n\r\n <!-- DEBUG: Verifica che formGroup sia inizializzato -->\r\n <!-- [formGroup] e sul div padre; tutti i formControlName INLINE funzionano -->\r\n\r\n <!-- ==================== HEADER ==================== -->\r\n @if (schema.title || schema.description) {\r\n <div class=\"ui-form-builder__header\">\r\n @if (schema.title) {\r\n <h2 class=\"ui-form-builder__title\">{{ schema.title }}</h2>\r\n }\r\n @if (schema.description) {\r\n <p class=\"ui-form-builder__description\">{{ schema.description }}</p>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- ==================== SEZIONI ==================== -->\r\n <div class=\"ui-form-builder__sections\">\r\n @for (section of schema.sections; track section.id) {\r\n @if (isSectionVisible(section.id)) {\r\n\r\n <!-- Sezione collassabile -->\r\n @if (section.collapsible) {\r\n <mat-expansion-panel\r\n class=\"ui-form-builder__section ui-form-builder__section--collapsible\"\r\n [class]=\"section.cssClasses?.join(' ')\"\r\n [expanded]=\"!section.collapsed\"\r\n >\r\n <mat-expansion-panel-header>\r\n @if (section.title) {\r\n <mat-panel-title>{{ section.title }}</mat-panel-title>\r\n }\r\n @if (section.description) {\r\n <mat-panel-description>{{ section.description }}</mat-panel-description>\r\n }\r\n </mat-expansion-panel-header>\r\n\r\n <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n @for (field of section.fields; track field.key) {\r\n @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n <div\r\n class=\"ui-form-builder__field-wrapper\"\r\n [class]=\"getFieldWrapperClasses(field)\"\r\n [style.order]=\"field.layout?.order || null\"\r\n [attr.data-field-key]=\"field.key\"\r\n >\r\n <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n <ng-container\r\n *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n />\r\n </div>\r\n }\r\n }\r\n </div>\r\n </mat-expansion-panel>\r\n } @else {\r\n <!-- Sezione non collassabile -->\r\n <div class=\"ui-form-builder__section\" [class]=\"section.cssClasses?.join(' ')\">\r\n @if (section.title || section.description) {\r\n <div class=\"ui-form-builder__section-header\">\r\n @if (section.title) {\r\n <h3 class=\"ui-form-builder__section-title\">{{ section.title }}</h3>\r\n }\r\n @if (section.description) {\r\n <p class=\"ui-form-builder__section-description\">{{ section.description }}</p>\r\n }\r\n </div>\r\n }\r\n\r\n <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n @for (field of section.fields; track field.key) {\r\n @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n <div\r\n class=\"ui-form-builder__field-wrapper\"\r\n [class]=\"getFieldWrapperClasses(field)\"\r\n [style.order]=\"field.layout?.order || null\"\r\n [attr.data-field-key]=\"field.key\"\r\n >\r\n <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n <ng-container\r\n *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n />\r\n </div>\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n }\r\n }\r\n </div>\r\n\r\n <!-- ==================== FOOTER ==================== -->\r\n @if (!schema.config?.hideFooter) {\r\n <div class=\"ui-form-builder__footer\">\r\n <ui-form-error-summary\r\n [errors]=\"formErrors\"\r\n [totalErrorCount]=\"getFormErrorsCount()\"\r\n (fieldClick)=\"scrollToField($event)\"\r\n />\r\n <ui-button-area\r\n [buttons]=\"formButtons\"\r\n align=\"end\"\r\n [loadingIds]=\"loadingFor\"\r\n />\r\n </div>\r\n }\r\n </div>\r\n}\r\n\r\n<!-- ============================================================ -->\r\n<!-- FIELD TEMPLATE -->\r\n<!-- ============================================================ -->\r\n<!--\r\n CRITICO: Questo template e dichiarato FUORI dal <div [formGroup]>.\r\n Angular risolve l'injector dal contesto di DICHIARAZIONE.\r\n Quindi DOBBIAMO wrappare il contenuto in un <div [formGroup]>\r\n per fornire un FormGroupDirective ai formControlName figli.\r\n Senza questo wrapper -> errore NG01050.\r\n-->\r\n<ng-template #fieldTpl let-field>\r\n @if (formGroup) {\r\n <div [formGroup]=\"formGroup\" class=\"ui-fb-field-ctx\">\r\n @switch (field.type) {\r\n\r\n <!-- TEXT / EMAIL / PASSWORD -->\r\n @case ('text') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n @case ('email') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n @case ('password') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n\r\n <!-- NUMBER -->\r\n @case ('number') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n @if (field.metadata?.type === 'currency') {\r\n <input matInput type=\"text\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n uiCurrencyInput\r\n [currencySymbol]=\"field.metadata?.currency || 'EUR'\"\r\n [decimalPlaces]=\"field.metadata?.decimals ?? 2\"\r\n />\r\n } @else {\r\n <input matInput type=\"number\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n }\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- TEXTAREA -->\r\n @case ('textarea') {\r\n <div class=\"ui-form-builder__textarea-wrapper\">\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <textarea matInput [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n rows=\"4\"\r\n ></textarea>\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n @if (getCharCount(field.key); as cc) {\r\n <span class=\"ui-form-builder__char-count\"\r\n [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n {{ cc.current }} / {{ cc.max }}\r\n </span>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- SELECT -->\r\n @case ('select') {\r\n @if (field.searchable) {\r\n <!-- Autocomplete select -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [matAutocomplete]=\"auto\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (input)=\"filterOptions(field, $any($event.target).value)\"\r\n />\r\n <mat-autocomplete #auto=\"matAutocomplete\" [displayWith]=\"displayFn(getFieldOptions(field))\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n }\r\n {{ opt.label }}\r\n </mat-option>\r\n }\r\n </mat-autocomplete>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n } @else {\r\n <!-- Standard select -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-select [formControlName]=\"field.key\" [placeholder]=\"field.placeholder || ''\">\r\n @if (!field.hideEmptyOption) {\r\n <mat-option [value]=\"null\">--</mat-option>\r\n }\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n }\r\n {{ opt.label }}\r\n @if (opt.tooltip) {\r\n <lucide-icon name=\"info\" [size]=\"14\"\r\n [matTooltip]=\"opt.tooltip\" class=\"ui-form-builder__option-info\" />\r\n }\r\n </mat-option>\r\n }\r\n </mat-select>\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n }\r\n\r\n <!-- MULTISELECT -->\r\n @case ('multiselect') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-select [formControlName]=\"field.key\" multiple [placeholder]=\"field.placeholder || ''\">\r\n @if (field.allowSelectAll) {\r\n <mat-option (click)=\"selectAll(field)\" (keydown.enter)=\"selectAll(field)\" (keydown.space)=\"selectAll(field)\">\r\n <em>Seleziona tutto</em>\r\n </mat-option>\r\n }\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n {{ opt.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n <!-- Chips preview sotto il select -->\r\n @if (formGroup.get(field.key)?.value?.length) {\r\n <div class=\"ui-form-builder__chips-preview\">\r\n @for (val of formGroup.get(field.key)!.value; track val) {\r\n <span class=\"ui-form-builder__chip\">\r\n {{ getOptionLabel(field, val) }}\r\n <button type=\"button\" (click)=\"removeMultiselectChip(field, val)\"\r\n class=\"ui-form-builder__chip-remove\" aria-label=\"Rimuovi\">\r\n <lucide-icon name=\"x\" [size]=\"12\" />\r\n </button>\r\n </span>\r\n }\r\n </div>\r\n }\r\n }\r\n\r\n <!-- FREEMULTISELECT -->\r\n @case ('freemultiselect') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width ui-form-builder__free-multi-field\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-chip-grid #chipGrid [formControlName]=\"field.key\">\r\n @for (chip of formGroup.get(field.key)?.value || []; track chip) {\r\n <mat-chip-row [removable]=\"!isChipProtected(field, chip)\" (removed)=\"removeFreeChip(field, chip)\">\r\n {{ chip }}\r\n @if (!isChipProtected(field, chip)) {\r\n <button matChipRemove aria-label=\"Rimuovi\">\r\n <lucide-icon name=\"x\" [size]=\"14\" />\r\n </button>\r\n }\r\n </mat-chip-row>\r\n }\r\n </mat-chip-grid>\r\n <input matInput\r\n [matChipInputFor]=\"chipGrid\"\r\n [placeholder]=\"field.placeholder || 'Aggiungi...'\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (matChipInputTokenEnd)=\"addFreeChip(field, $event)\"\r\n (blur)=\"addFreeChipOnBlur(field, $event)\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- CHECKBOX -->\r\n @case ('checkbox') {\r\n @if (field.appearance?.style === 'switch') {\r\n <mat-slide-toggle\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n </mat-slide-toggle>\r\n } @else {\r\n <mat-checkbox\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n @if (field.iconTooltip) {\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n }\r\n </mat-checkbox>\r\n }\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n }\r\n\r\n <!-- SWITCH -->\r\n @case ('switch') {\r\n <mat-slide-toggle\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n @if (field.iconTooltip) {\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n }\r\n </mat-slide-toggle>\r\n }\r\n\r\n <!-- RADIO -->\r\n @case ('radio') {\r\n <div class=\"ui-form-builder__radio-wrapper\">\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n <mat-radio-group [formControlName]=\"field.key\" [color]=\"field.appearance?.color || 'primary'\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-radio-button [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" />\r\n }\r\n {{ opt.label }}\r\n </mat-radio-button>\r\n }\r\n </mat-radio-group>\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- DATE -->\r\n @case ('date') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [matDatepicker]=\"datepicker\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || 'GG/MM/AAAA'\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n <mat-datepicker-toggle matSuffix [for]=\"datepicker\" />\r\n <mat-datepicker #datepicker\r\n [startView]=\"field.customConfig?.config?.['monthView'] ? 'year' : 'month'\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- DATETIME -->\r\n @case ('datetime') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput type=\"datetime-local\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- FILE -->\r\n @case ('file') {\r\n <div class=\"ui-form-builder__file-wrapper\">\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n <ui-file-input\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.fileConfig\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- DIVIDER -->\r\n @case ('divider') {\r\n <div class=\"ui-form-builder__divider\">\r\n @if (field.label) {\r\n <span class=\"ui-form-builder__divider-label\">{{ field.label }}</span>\r\n }\r\n <hr class=\"ui-form-builder__divider-line\" />\r\n </div>\r\n }\r\n\r\n <!-- LOCATION (selezione territorio) -->\r\n @case ('location') {\r\n <div class=\"ui-form-builder__location-wrapper\">\r\n @if (field.label) {\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n }\r\n <ui-specifica-territoriale\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.customConfig?.config || {}\"\r\n [disabled]=\"readonly || field.disabled || disabled\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- LOCATION-TABLE (tabella CRUD di location) -->\r\n @case ('location-table') {\r\n <div class=\"ui-form-builder__location-table-wrapper\">\r\n <ui-table-territoriale\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.customConfig?.config || {}\"\r\n [disabled]=\"readonly || field.disabled || disabled\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- CUSTOM -->\r\n @case ('custom') {\r\n <div class=\"ui-form-builder__custom-wrapper\" [attr.data-component]=\"field.customConfig?.component\">\r\n @if (field.label) {\r\n <label class=\"ui-form-builder__field-label\">{{ field.label }}</label>\r\n }\r\n <!-- I componenti custom devono essere proiettati dall'applicazione host -->\r\n <div class=\"ui-form-builder__custom-placeholder\">\r\n <span>Componente custom: {{ field.customConfig?.component }}</span>\r\n </div>\r\n </div>\r\n }\r\n }\r\n </div>\r\n }\r\n</ng-template>\r\n\r\n<!-- ============================================================ -->\r\n<!-- TEXT / EMAIL / PASSWORD template -->\r\n<!-- ============================================================ -->\r\n<!--\r\n CRITICO: Anche questo template necessita del wrapper [formGroup].\r\n Viene invocato da fieldTpl che gia ha il wrapper, ma siccome\r\n questo e un SECONDO ng-template, Angular crea un NUOVO contesto\r\n di dichiarazione. Il wrapper di fieldTpl non e sufficiente.\r\n-->\r\n<ng-template #textFieldTpl let-field>\r\n @if (formGroup) {\r\n <div [formGroup]=\"formGroup\" class=\"ui-fb-text-ctx\">\r\n @if (field.searchable && field.type === 'text') {\r\n <!-- Autocomplete text -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [type]=\"getNativeInputType(field)\"\r\n [matAutocomplete]=\"textAuto\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (input)=\"filterOptions(field, $any($event.target).value)\"\r\n />\r\n <mat-autocomplete #textAuto=\"matAutocomplete\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\">{{ opt.label }}</mat-option>\r\n }\r\n </mat-autocomplete>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n } @else {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [type]=\"getNativeInputType(field)\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n @if (field.type === 'text' && getCharCount(field.key); as cc) {\r\n <span class=\"ui-form-builder__char-count\"\r\n [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n {{ cc.current }} / {{ cc.max }}\r\n </span>\r\n }\r\n }\r\n </div>\r\n }\r\n</ng-template>\r\n", styles: [".ui-form-builder{display:flex;flex-direction:column;gap:var(--ui-spacing-6);font-family:var(--ui-font-family)}.ui-form-builder__header{margin-bottom:var(--ui-spacing-2)}.ui-form-builder__title{font-size:var(--ui-font-size-xl);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);margin:0 0 var(--ui-spacing-1)}.ui-form-builder__description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0}.ui-form-builder__sections{display:flex;flex-direction:column;gap:var(--ui-spacing-6)}.ui-form-builder__section--collapsible .mat-expansion-panel-body{padding:var(--ui-spacing-4) var(--ui-spacing-3) var(--ui-spacing-3)}.ui-form-builder__section-header{margin-bottom:var(--ui-spacing-4);padding-bottom:var(--ui-spacing-2);border-bottom:1px solid var(--ui-color-border)}.ui-form-builder__section-title{font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);margin:0 0 var(--ui-spacing-1)}.ui-form-builder__section-description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0}.ui-form-builder__grid{display:grid;grid-template-columns:repeat(12,1fr);gap:var(--ui-spacing-4) var(--ui-spacing-4);align-items:start}.ui-form-builder__field-wrapper{grid-column:span 12;min-width:0}@media (min-width: 768px){.ui-form-builder__field-wrapper{grid-column:span 6}}@media (min-width: 576px){.ui-form-builder__field-wrapper.ui-col-sm-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-sm-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-sm-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-sm-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-sm-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-sm-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-sm-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-sm-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-sm-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-sm-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-sm-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-sm-12{grid-column:span 12}}@media (min-width: 768px){.ui-form-builder__field-wrapper.ui-col-md-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-md-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-md-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-md-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-md-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-md-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-md-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-md-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-md-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-md-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-md-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-md-12{grid-column:span 12}}@media (min-width: 1024px){.ui-form-builder__field-wrapper.ui-col-lg-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-lg-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-lg-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-lg-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-lg-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-lg-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-lg-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-lg-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-lg-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-lg-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-lg-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-lg-12{grid-column:span 12}}@media (min-width: 1280px){.ui-form-builder__field-wrapper.ui-col-xl-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-xl-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-xl-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-xl-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-xl-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-xl-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-xl-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-xl-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-xl-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-xl-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-xl-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-xl-12{grid-column:span 12}}.ui-form-builder__full-width{width:100%}.ui-fb-field-ctx,.ui-fb-text-ctx{display:contents}.ui-form-builder__textarea-wrapper{position:relative}.ui-form-builder__char-count{display:block;text-align:right;font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);margin-top:calc(-1 * var(--ui-spacing-3))}.ui-form-builder__char-count--warn{color:var(--ui-color-warn, #d97706)}.ui-form-builder__char-count--error{color:var(--ui-color-error, #dc2626);font-weight:var(--ui-font-weight-semibold)}.ui-form-builder__field-label{display:block;font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text);margin-bottom:var(--ui-spacing-4)}.ui-form-builder__required{color:var(--ui-color-error, #dc2626);margin-left:2px}.ui-form-builder__field-error{margin-top:var(--ui-spacing-1);font-size:var(--ui-font-size-xs);color:var(--ui-color-error, #dc2626);display:flex;flex-direction:column;gap:2px}.ui-form-builder__radio-wrapper mat-radio-group{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-3)}.ui-form-builder__chips-preview{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-1);margin-top:var(--ui-spacing-1)}.ui-form-builder__chip{display:inline-flex;align-items:center;gap:4px;padding:2px var(--ui-spacing-2);border-radius:var(--ui-radius-full);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .08));color:var(--ui-color-primary);font-size:var(--ui-font-size-xs)}.ui-form-builder__chip-remove{appearance:none;border:none;background:transparent;cursor:pointer;padding:0;display:flex;color:inherit;opacity:.7}.ui-form-builder__chip-remove:hover{opacity:1}.ui-form-builder__option-icon{margin-right:var(--ui-spacing-2);vertical-align:middle}.ui-form-builder__option-info{margin-left:auto;color:var(--ui-color-text-muted)}.ui-form-builder__inline-tooltip{margin-left:var(--ui-spacing-1);color:var(--ui-color-text-muted);cursor:help}.ui-form-builder__divider{grid-column:1/-1!important;display:flex;align-items:center;gap:var(--ui-spacing-3);padding:var(--ui-spacing-2) 0}.ui-form-builder__divider-label{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text-secondary);white-space:nowrap}.ui-form-builder__divider-line{flex:1;border:none;border-top:1px solid var(--ui-color-border);margin:0}.ui-form-builder__custom-placeholder{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-4);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-md);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm)}.ui-form-builder__footer{display:flex;align-items:center;justify-content:space-between;gap:var(--ui-spacing-4);padding-top:var(--ui-spacing-4);border-top:1px solid var(--ui-color-border)}@media (max-width: 767.98px){.ui-form-builder__footer{flex-direction:column;align-items:stretch}}.ui-form-field--highlight{animation:ui-field-highlight-pulse 1s ease-in-out 2;border-radius:var(--ui-radius-md)}@keyframes ui-field-highlight-pulse{0%{box-shadow:0 0 #dc262666;padding:10px}50%{box-shadow:0 0 0 6px #dc262626;padding:10px}to{box-shadow:0 0 #dc262600;padding:10px}}.ui-form-builder mat-error{font-size:var(--ui-font-size-xs, 12px);display:flex;flex-direction:column;transform:translateY(-5px)}.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__leading,.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__notch,.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__trailing{border-color:var(--ui-color-error, #dc2626)!important}.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mat-mdc-floating-label{color:var(--ui-color-error, #dc2626)}.ui-form-builder__section--collapsible.mat-expansion-panel{box-shadow:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg)!important}.ui-form-builder__free-multi-field mat-chip-grid{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-1)}.ui-form-builder__free-multi-field input[matInput]{flex:1;min-width:80px}.ui-form-builder mat-datepicker-toggle{display:inline-flex;align-items:center}.mdc-notched-outline__notch{border-left:none!important}.mat-expansion-panel{box-shadow:none!important}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.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: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "directive", type: i2.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i2.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.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: "ngmodule", type: MatSelectModule }, { kind: "component", type: i4.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i5.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "component", type: i6.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "component", type: i7.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: MatRadioModule }, { kind: "directive", type: i8.MatRadioGroup, selector: "mat-radio-group", inputs: ["color", "name", "labelPosition", "value", "selected", "disabled", "required", "disabledInteractive"], outputs: ["change"], exportAs: ["matRadioGroup"] }, { kind: "component", type: i8.MatRadioButton, selector: "mat-radio-button", inputs: ["id", "name", "aria-label", "aria-labelledby", "aria-describedby", "disableRipple", "tabIndex", "checked", "value", "labelPosition", "disabled", "required", "color", "disabledInteractive"], outputs: ["change"], exportAs: ["matRadioButton"] }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "component", type: i9.MatDatepicker, selector: "mat-datepicker", exportAs: ["matDatepicker"] }, { kind: "directive", type: i9.MatDatepickerInput, selector: "input[matDatepicker]", inputs: ["matDatepicker", "min", "max", "matDatepickerFilter"], exportAs: ["matDatepickerInput"] }, { kind: "component", type: i9.MatDatepickerToggle, selector: "mat-datepicker-toggle", inputs: ["for", "tabIndex", "aria-label", "disabled", "disableRipple"], exportAs: ["matDatepickerToggle"] }, { kind: "ngmodule", type: MatNativeDateModule }, { kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i10.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "directive", type: i10.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: MatChipsModule }, { kind: "component", type: i11.MatChipGrid, selector: "mat-chip-grid", inputs: ["disabled", "placeholder", "required", "value", "errorStateMatcher"], outputs: ["change", "valueChange"] }, { kind: "directive", type: i11.MatChipInput, selector: "input[matChipInputFor]", inputs: ["matChipInputFor", "matChipInputAddOnBlur", "matChipInputSeparatorKeyCodes", "placeholder", "id", "disabled"], outputs: ["matChipInputTokenEnd"], exportAs: ["matChipInput", "matChipInputFor"] }, { kind: "directive", type: i11.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i11.MatChipRow, selector: "mat-chip-row, [mat-chip-row], mat-basic-chip-row, [mat-basic-chip-row]", inputs: ["editable"], outputs: ["edited"] }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "component", type: i12.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i12.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i12.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "directive", type: i12.MatExpansionPanelDescription, selector: "mat-panel-description" }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i13.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i14.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonAreaComponent, selector: "ui-button-area", inputs: ["buttons", "align", "ariaLabel", "gap", "stackOnMobile", "disableWhileLoading", "loadingIds"] }, { kind: "component", type: UiFormErrorSummaryComponent, selector: "ui-form-error-summary", inputs: ["errors", "totalErrorCount"], outputs: ["fieldClick"] }, { kind: "component", type: UiFileInputComponent, selector: "ui-file-input", inputs: ["config"], outputs: ["fileRemoved", "validationError"] }, { kind: "component", type: UiSpecificaTerritorialeComponent, selector: "ui-specifica-territoriale", inputs: ["config", "disabled"] }, { kind: "component", type: UiTableTerritorialeComponent, selector: "ui-table-territoriale", inputs: ["config", "disabled"] }, { kind: "directive", type: UiCurrencyInputDirective, selector: "[uiCurrencyInput]", inputs: ["currencySymbol", "decimalPlaces"] }], encapsulation: i0.ViewEncapsulation.None }); }
|
|
767
|
+
}
|
|
768
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormBuilderComponent, decorators: [{
|
|
769
|
+
type: Component,
|
|
770
|
+
args: [{ selector: 'ui-form-builder', standalone: true, imports: [
|
|
771
|
+
ReactiveFormsModule,
|
|
772
|
+
NgTemplateOutlet,
|
|
773
|
+
MatFormFieldModule,
|
|
774
|
+
MatInputModule,
|
|
775
|
+
MatSelectModule,
|
|
776
|
+
MatCheckboxModule,
|
|
777
|
+
MatSlideToggleModule,
|
|
778
|
+
MatRadioModule,
|
|
779
|
+
MatDatepickerModule,
|
|
780
|
+
MatNativeDateModule,
|
|
781
|
+
MatAutocompleteModule,
|
|
782
|
+
MatChipsModule,
|
|
783
|
+
MatExpansionModule,
|
|
784
|
+
MatTooltipModule,
|
|
785
|
+
LucideAngularModule,
|
|
786
|
+
UiButtonAreaComponent,
|
|
787
|
+
UiFormErrorSummaryComponent,
|
|
788
|
+
UiFileInputComponent,
|
|
789
|
+
UiSpecificaTerritorialeComponent,
|
|
790
|
+
UiTableTerritorialeComponent,
|
|
791
|
+
UiCurrencyInputDirective,
|
|
792
|
+
], providers: [
|
|
793
|
+
{ provide: ErrorStateMatcher, useClass: UiFormErrorStateMatcher },
|
|
794
|
+
// Locale italiano per il datepicker Material
|
|
795
|
+
{ provide: MAT_DATE_LOCALE, useValue: 'it-IT' },
|
|
796
|
+
// Adapter personalizzato per parsing DD/MM/YYYY
|
|
797
|
+
{ provide: DateAdapter, useClass: UiItalianDateAdapter },
|
|
798
|
+
// Formati data italiani (DD/MM/YYYY)
|
|
799
|
+
{ provide: MAT_DATE_FORMATS, useValue: UI_IT_DATE_FORMATS },
|
|
800
|
+
], encapsulation: ViewEncapsulation.None, host: { class: 'ui-form-builder-host' }, template: "<!--\r\n ============================================================\r\n UI FORM BUILDER - TEMPLATE PRINCIPALE\r\n ============================================================\r\n ARCHITETTURA:\r\n Ogni <ng-template> che contiene direttive [formControlName]\r\n DEVE wrappare il proprio contenuto in un <div [formGroup]>.\r\n Questo perche Angular risolve l'injector di formControlName\r\n dal CONTESTO DI DICHIARAZIONE del template, non dal punto\r\n di inserimento (ngTemplateOutlet).\r\n\r\n Senza questo wrapper, formControlName non trova il\r\n FormGroupDirective e lancia NG01050.\r\n\r\n Ref: versione tailored (ang-ms-webapp) usa lo stesso pattern.\r\n ============================================================\r\n-->\r\n\r\n@if (schema && formGroup) {\r\n <div class=\"ui-form-builder\" [formGroup]=\"formGroup\" [class]=\"schema.config?.cssClasses?.join(' ')\">\r\n\r\n <!-- DEBUG: Verifica che formGroup sia inizializzato -->\r\n <!-- [formGroup] e sul div padre; tutti i formControlName INLINE funzionano -->\r\n\r\n <!-- ==================== HEADER ==================== -->\r\n @if (schema.title || schema.description) {\r\n <div class=\"ui-form-builder__header\">\r\n @if (schema.title) {\r\n <h2 class=\"ui-form-builder__title\">{{ schema.title }}</h2>\r\n }\r\n @if (schema.description) {\r\n <p class=\"ui-form-builder__description\">{{ schema.description }}</p>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- ==================== SEZIONI ==================== -->\r\n <div class=\"ui-form-builder__sections\">\r\n @for (section of schema.sections; track section.id) {\r\n @if (isSectionVisible(section.id)) {\r\n\r\n <!-- Sezione collassabile -->\r\n @if (section.collapsible) {\r\n <mat-expansion-panel\r\n class=\"ui-form-builder__section ui-form-builder__section--collapsible\"\r\n [class]=\"section.cssClasses?.join(' ')\"\r\n [expanded]=\"!section.collapsed\"\r\n >\r\n <mat-expansion-panel-header>\r\n @if (section.title) {\r\n <mat-panel-title>{{ section.title }}</mat-panel-title>\r\n }\r\n @if (section.description) {\r\n <mat-panel-description>{{ section.description }}</mat-panel-description>\r\n }\r\n </mat-expansion-panel-header>\r\n\r\n <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n @for (field of section.fields; track field.key) {\r\n @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n <div\r\n class=\"ui-form-builder__field-wrapper\"\r\n [class]=\"getFieldWrapperClasses(field)\"\r\n [style.order]=\"field.layout?.order || null\"\r\n [attr.data-field-key]=\"field.key\"\r\n >\r\n <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n <ng-container\r\n *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n />\r\n </div>\r\n }\r\n }\r\n </div>\r\n </mat-expansion-panel>\r\n } @else {\r\n <!-- Sezione non collassabile -->\r\n <div class=\"ui-form-builder__section\" [class]=\"section.cssClasses?.join(' ')\">\r\n @if (section.title || section.description) {\r\n <div class=\"ui-form-builder__section-header\">\r\n @if (section.title) {\r\n <h3 class=\"ui-form-builder__section-title\">{{ section.title }}</h3>\r\n }\r\n @if (section.description) {\r\n <p class=\"ui-form-builder__section-description\">{{ section.description }}</p>\r\n }\r\n </div>\r\n }\r\n\r\n <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n @for (field of section.fields; track field.key) {\r\n @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n <div\r\n class=\"ui-form-builder__field-wrapper\"\r\n [class]=\"getFieldWrapperClasses(field)\"\r\n [style.order]=\"field.layout?.order || null\"\r\n [attr.data-field-key]=\"field.key\"\r\n >\r\n <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n <ng-container\r\n *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n />\r\n </div>\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n }\r\n }\r\n </div>\r\n\r\n <!-- ==================== FOOTER ==================== -->\r\n @if (!schema.config?.hideFooter) {\r\n <div class=\"ui-form-builder__footer\">\r\n <ui-form-error-summary\r\n [errors]=\"formErrors\"\r\n [totalErrorCount]=\"getFormErrorsCount()\"\r\n (fieldClick)=\"scrollToField($event)\"\r\n />\r\n <ui-button-area\r\n [buttons]=\"formButtons\"\r\n align=\"end\"\r\n [loadingIds]=\"loadingFor\"\r\n />\r\n </div>\r\n }\r\n </div>\r\n}\r\n\r\n<!-- ============================================================ -->\r\n<!-- FIELD TEMPLATE -->\r\n<!-- ============================================================ -->\r\n<!--\r\n CRITICO: Questo template e dichiarato FUORI dal <div [formGroup]>.\r\n Angular risolve l'injector dal contesto di DICHIARAZIONE.\r\n Quindi DOBBIAMO wrappare il contenuto in un <div [formGroup]>\r\n per fornire un FormGroupDirective ai formControlName figli.\r\n Senza questo wrapper -> errore NG01050.\r\n-->\r\n<ng-template #fieldTpl let-field>\r\n @if (formGroup) {\r\n <div [formGroup]=\"formGroup\" class=\"ui-fb-field-ctx\">\r\n @switch (field.type) {\r\n\r\n <!-- TEXT / EMAIL / PASSWORD -->\r\n @case ('text') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n @case ('email') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n @case ('password') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n\r\n <!-- NUMBER -->\r\n @case ('number') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n @if (field.metadata?.type === 'currency') {\r\n <input matInput type=\"text\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n uiCurrencyInput\r\n [currencySymbol]=\"field.metadata?.currency || 'EUR'\"\r\n [decimalPlaces]=\"field.metadata?.decimals ?? 2\"\r\n />\r\n } @else {\r\n <input matInput type=\"number\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n }\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- TEXTAREA -->\r\n @case ('textarea') {\r\n <div class=\"ui-form-builder__textarea-wrapper\">\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <textarea matInput [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n rows=\"4\"\r\n ></textarea>\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n @if (getCharCount(field.key); as cc) {\r\n <span class=\"ui-form-builder__char-count\"\r\n [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n {{ cc.current }} / {{ cc.max }}\r\n </span>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- SELECT -->\r\n @case ('select') {\r\n @if (field.searchable) {\r\n <!-- Autocomplete select -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [matAutocomplete]=\"auto\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (input)=\"filterOptions(field, $any($event.target).value)\"\r\n />\r\n <mat-autocomplete #auto=\"matAutocomplete\" [displayWith]=\"displayFn(getFieldOptions(field))\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n }\r\n {{ opt.label }}\r\n </mat-option>\r\n }\r\n </mat-autocomplete>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n } @else {\r\n <!-- Standard select -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-select [formControlName]=\"field.key\" [placeholder]=\"field.placeholder || ''\">\r\n @if (!field.hideEmptyOption) {\r\n <mat-option [value]=\"null\">--</mat-option>\r\n }\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n }\r\n {{ opt.label }}\r\n @if (opt.tooltip) {\r\n <lucide-icon name=\"info\" [size]=\"14\"\r\n [matTooltip]=\"opt.tooltip\" class=\"ui-form-builder__option-info\" />\r\n }\r\n </mat-option>\r\n }\r\n </mat-select>\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n }\r\n\r\n <!-- MULTISELECT -->\r\n @case ('multiselect') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-select [formControlName]=\"field.key\" multiple [placeholder]=\"field.placeholder || ''\">\r\n @if (field.allowSelectAll) {\r\n <mat-option (click)=\"selectAll(field)\" (keydown.enter)=\"selectAll(field)\" (keydown.space)=\"selectAll(field)\">\r\n <em>Seleziona tutto</em>\r\n </mat-option>\r\n }\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n {{ opt.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n <!-- Chips preview sotto il select -->\r\n @if (formGroup.get(field.key)?.value?.length) {\r\n <div class=\"ui-form-builder__chips-preview\">\r\n @for (val of formGroup.get(field.key)!.value; track val) {\r\n <span class=\"ui-form-builder__chip\">\r\n {{ getOptionLabel(field, val) }}\r\n <button type=\"button\" (click)=\"removeMultiselectChip(field, val)\"\r\n class=\"ui-form-builder__chip-remove\" aria-label=\"Rimuovi\">\r\n <lucide-icon name=\"x\" [size]=\"12\" />\r\n </button>\r\n </span>\r\n }\r\n </div>\r\n }\r\n }\r\n\r\n <!-- FREEMULTISELECT -->\r\n @case ('freemultiselect') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width ui-form-builder__free-multi-field\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <mat-chip-grid #chipGrid [formControlName]=\"field.key\">\r\n @for (chip of formGroup.get(field.key)?.value || []; track chip) {\r\n <mat-chip-row [removable]=\"!isChipProtected(field, chip)\" (removed)=\"removeFreeChip(field, chip)\">\r\n {{ chip }}\r\n @if (!isChipProtected(field, chip)) {\r\n <button matChipRemove aria-label=\"Rimuovi\">\r\n <lucide-icon name=\"x\" [size]=\"14\" />\r\n </button>\r\n }\r\n </mat-chip-row>\r\n }\r\n </mat-chip-grid>\r\n <input matInput\r\n [matChipInputFor]=\"chipGrid\"\r\n [placeholder]=\"field.placeholder || 'Aggiungi...'\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (matChipInputTokenEnd)=\"addFreeChip(field, $event)\"\r\n (blur)=\"addFreeChipOnBlur(field, $event)\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- CHECKBOX -->\r\n @case ('checkbox') {\r\n @if (field.appearance?.style === 'switch') {\r\n <mat-slide-toggle\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n </mat-slide-toggle>\r\n } @else {\r\n <mat-checkbox\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n @if (field.iconTooltip) {\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n }\r\n </mat-checkbox>\r\n }\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n }\r\n\r\n <!-- SWITCH -->\r\n @case ('switch') {\r\n <mat-slide-toggle\r\n [formControlName]=\"field.key\"\r\n [color]=\"field.appearance?.color || 'primary'\"\r\n >\r\n {{ field.label }}\r\n @if (field.iconTooltip) {\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n }\r\n </mat-slide-toggle>\r\n }\r\n\r\n <!-- RADIO -->\r\n @case ('radio') {\r\n <div class=\"ui-form-builder__radio-wrapper\">\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n <mat-radio-group [formControlName]=\"field.key\" [color]=\"field.appearance?.color || 'primary'\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-radio-button [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n @if (opt.icon) {\r\n <lucide-icon [name]=\"opt.icon\" [size]=\"16\" />\r\n }\r\n {{ opt.label }}\r\n </mat-radio-button>\r\n }\r\n </mat-radio-group>\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- DATE -->\r\n @case ('date') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [matDatepicker]=\"datepicker\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || 'GG/MM/AAAA'\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n <mat-datepicker-toggle matSuffix [for]=\"datepicker\" />\r\n <mat-datepicker #datepicker\r\n [startView]=\"field.customConfig?.config?.['monthView'] ? 'year' : 'month'\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- DATETIME -->\r\n @case ('datetime') {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput type=\"datetime-local\" [formControlName]=\"field.key\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- FILE -->\r\n @case ('file') {\r\n <div class=\"ui-form-builder__file-wrapper\">\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n <ui-file-input\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.fileConfig\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- DIVIDER -->\r\n @case ('divider') {\r\n <div class=\"ui-form-builder__divider\">\r\n @if (field.label) {\r\n <span class=\"ui-form-builder__divider-label\">{{ field.label }}</span>\r\n }\r\n <hr class=\"ui-form-builder__divider-line\" />\r\n </div>\r\n }\r\n\r\n <!-- LOCATION (selezione territorio) -->\r\n @case ('location') {\r\n <div class=\"ui-form-builder__location-wrapper\">\r\n @if (field.label) {\r\n <label class=\"ui-form-builder__field-label\">\r\n {{ field.label }}\r\n @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n </label>\r\n }\r\n <ui-specifica-territoriale\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.customConfig?.config || {}\"\r\n [disabled]=\"readonly || field.disabled || disabled\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- LOCATION-TABLE (tabella CRUD di location) -->\r\n @case ('location-table') {\r\n <div class=\"ui-form-builder__location-table-wrapper\">\r\n <ui-table-territoriale\r\n [formControlName]=\"field.key\"\r\n [config]=\"field.customConfig?.config || {}\"\r\n [disabled]=\"readonly || field.disabled || disabled\"\r\n />\r\n @if (shouldShowFieldErrors(field.key)) {\r\n <div class=\"ui-form-builder__field-error\">\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <span>{{ err }}</span>\r\n }\r\n </div>\r\n }\r\n </div>\r\n }\r\n\r\n <!-- CUSTOM -->\r\n @case ('custom') {\r\n <div class=\"ui-form-builder__custom-wrapper\" [attr.data-component]=\"field.customConfig?.component\">\r\n @if (field.label) {\r\n <label class=\"ui-form-builder__field-label\">{{ field.label }}</label>\r\n }\r\n <!-- I componenti custom devono essere proiettati dall'applicazione host -->\r\n <div class=\"ui-form-builder__custom-placeholder\">\r\n <span>Componente custom: {{ field.customConfig?.component }}</span>\r\n </div>\r\n </div>\r\n }\r\n }\r\n </div>\r\n }\r\n</ng-template>\r\n\r\n<!-- ============================================================ -->\r\n<!-- TEXT / EMAIL / PASSWORD template -->\r\n<!-- ============================================================ -->\r\n<!--\r\n CRITICO: Anche questo template necessita del wrapper [formGroup].\r\n Viene invocato da fieldTpl che gia ha il wrapper, ma siccome\r\n questo e un SECONDO ng-template, Angular crea un NUOVO contesto\r\n di dichiarazione. Il wrapper di fieldTpl non e sufficiente.\r\n-->\r\n<ng-template #textFieldTpl let-field>\r\n @if (formGroup) {\r\n <div [formGroup]=\"formGroup\" class=\"ui-fb-text-ctx\">\r\n @if (field.searchable && field.type === 'text') {\r\n <!-- Autocomplete text -->\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [type]=\"getNativeInputType(field)\"\r\n [matAutocomplete]=\"textAuto\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n (input)=\"filterOptions(field, $any($event.target).value)\"\r\n />\r\n <mat-autocomplete #textAuto=\"matAutocomplete\">\r\n @for (opt of getFieldOptions(field); track opt.value) {\r\n <mat-option [value]=\"opt.value\">{{ opt.label }}</mat-option>\r\n }\r\n </mat-autocomplete>\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n } @else {\r\n <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n <mat-label>{{ field.label }}</mat-label>\r\n <input matInput [formControlName]=\"field.key\"\r\n [type]=\"getNativeInputType(field)\"\r\n [placeholder]=\"field.placeholder || ''\"\r\n [readonly]=\"field.readonly || readonly\"\r\n />\r\n @if (field.iconTooltip) {\r\n <button matSuffix mat-icon-button type=\"button\"\r\n [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n </button>\r\n }\r\n <mat-error>\r\n @for (err of getFieldErrors(field.key); track $index) {\r\n <div>{{ err }}</div>\r\n }\r\n </mat-error>\r\n </mat-form-field>\r\n @if (field.type === 'text' && getCharCount(field.key); as cc) {\r\n <span class=\"ui-form-builder__char-count\"\r\n [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n {{ cc.current }} / {{ cc.max }}\r\n </span>\r\n }\r\n }\r\n </div>\r\n }\r\n</ng-template>\r\n", styles: [".ui-form-builder{display:flex;flex-direction:column;gap:var(--ui-spacing-6);font-family:var(--ui-font-family)}.ui-form-builder__header{margin-bottom:var(--ui-spacing-2)}.ui-form-builder__title{font-size:var(--ui-font-size-xl);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);margin:0 0 var(--ui-spacing-1)}.ui-form-builder__description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0}.ui-form-builder__sections{display:flex;flex-direction:column;gap:var(--ui-spacing-6)}.ui-form-builder__section--collapsible .mat-expansion-panel-body{padding:var(--ui-spacing-4) var(--ui-spacing-3) var(--ui-spacing-3)}.ui-form-builder__section-header{margin-bottom:var(--ui-spacing-4);padding-bottom:var(--ui-spacing-2);border-bottom:1px solid var(--ui-color-border)}.ui-form-builder__section-title{font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);margin:0 0 var(--ui-spacing-1)}.ui-form-builder__section-description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0}.ui-form-builder__grid{display:grid;grid-template-columns:repeat(12,1fr);gap:var(--ui-spacing-4) var(--ui-spacing-4);align-items:start}.ui-form-builder__field-wrapper{grid-column:span 12;min-width:0}@media (min-width: 768px){.ui-form-builder__field-wrapper{grid-column:span 6}}@media (min-width: 576px){.ui-form-builder__field-wrapper.ui-col-sm-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-sm-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-sm-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-sm-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-sm-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-sm-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-sm-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-sm-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-sm-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-sm-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-sm-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-sm-12{grid-column:span 12}}@media (min-width: 768px){.ui-form-builder__field-wrapper.ui-col-md-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-md-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-md-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-md-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-md-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-md-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-md-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-md-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-md-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-md-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-md-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-md-12{grid-column:span 12}}@media (min-width: 1024px){.ui-form-builder__field-wrapper.ui-col-lg-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-lg-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-lg-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-lg-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-lg-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-lg-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-lg-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-lg-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-lg-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-lg-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-lg-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-lg-12{grid-column:span 12}}@media (min-width: 1280px){.ui-form-builder__field-wrapper.ui-col-xl-1{grid-column:span 1}.ui-form-builder__field-wrapper.ui-col-xl-2{grid-column:span 2}.ui-form-builder__field-wrapper.ui-col-xl-3{grid-column:span 3}.ui-form-builder__field-wrapper.ui-col-xl-4{grid-column:span 4}.ui-form-builder__field-wrapper.ui-col-xl-5{grid-column:span 5}.ui-form-builder__field-wrapper.ui-col-xl-6{grid-column:span 6}.ui-form-builder__field-wrapper.ui-col-xl-7{grid-column:span 7}.ui-form-builder__field-wrapper.ui-col-xl-8{grid-column:span 8}.ui-form-builder__field-wrapper.ui-col-xl-9{grid-column:span 9}.ui-form-builder__field-wrapper.ui-col-xl-10{grid-column:span 10}.ui-form-builder__field-wrapper.ui-col-xl-11{grid-column:span 11}.ui-form-builder__field-wrapper.ui-col-xl-12{grid-column:span 12}}.ui-form-builder__full-width{width:100%}.ui-fb-field-ctx,.ui-fb-text-ctx{display:contents}.ui-form-builder__textarea-wrapper{position:relative}.ui-form-builder__char-count{display:block;text-align:right;font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);margin-top:calc(-1 * var(--ui-spacing-3))}.ui-form-builder__char-count--warn{color:var(--ui-color-warn, #d97706)}.ui-form-builder__char-count--error{color:var(--ui-color-error, #dc2626);font-weight:var(--ui-font-weight-semibold)}.ui-form-builder__field-label{display:block;font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text);margin-bottom:var(--ui-spacing-4)}.ui-form-builder__required{color:var(--ui-color-error, #dc2626);margin-left:2px}.ui-form-builder__field-error{margin-top:var(--ui-spacing-1);font-size:var(--ui-font-size-xs);color:var(--ui-color-error, #dc2626);display:flex;flex-direction:column;gap:2px}.ui-form-builder__radio-wrapper mat-radio-group{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-3)}.ui-form-builder__chips-preview{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-1);margin-top:var(--ui-spacing-1)}.ui-form-builder__chip{display:inline-flex;align-items:center;gap:4px;padding:2px var(--ui-spacing-2);border-radius:var(--ui-radius-full);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .08));color:var(--ui-color-primary);font-size:var(--ui-font-size-xs)}.ui-form-builder__chip-remove{appearance:none;border:none;background:transparent;cursor:pointer;padding:0;display:flex;color:inherit;opacity:.7}.ui-form-builder__chip-remove:hover{opacity:1}.ui-form-builder__option-icon{margin-right:var(--ui-spacing-2);vertical-align:middle}.ui-form-builder__option-info{margin-left:auto;color:var(--ui-color-text-muted)}.ui-form-builder__inline-tooltip{margin-left:var(--ui-spacing-1);color:var(--ui-color-text-muted);cursor:help}.ui-form-builder__divider{grid-column:1/-1!important;display:flex;align-items:center;gap:var(--ui-spacing-3);padding:var(--ui-spacing-2) 0}.ui-form-builder__divider-label{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text-secondary);white-space:nowrap}.ui-form-builder__divider-line{flex:1;border:none;border-top:1px solid var(--ui-color-border);margin:0}.ui-form-builder__custom-placeholder{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-4);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-md);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm)}.ui-form-builder__footer{display:flex;align-items:center;justify-content:space-between;gap:var(--ui-spacing-4);padding-top:var(--ui-spacing-4);border-top:1px solid var(--ui-color-border)}@media (max-width: 767.98px){.ui-form-builder__footer{flex-direction:column;align-items:stretch}}.ui-form-field--highlight{animation:ui-field-highlight-pulse 1s ease-in-out 2;border-radius:var(--ui-radius-md)}@keyframes ui-field-highlight-pulse{0%{box-shadow:0 0 #dc262666;padding:10px}50%{box-shadow:0 0 0 6px #dc262626;padding:10px}to{box-shadow:0 0 #dc262600;padding:10px}}.ui-form-builder mat-error{font-size:var(--ui-font-size-xs, 12px);display:flex;flex-direction:column;transform:translateY(-5px)}.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__leading,.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__notch,.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mdc-notched-outline__trailing{border-color:var(--ui-color-error, #dc2626)!important}.ui-form-builder .mat-mdc-form-field.mat-form-field-invalid .mat-mdc-floating-label{color:var(--ui-color-error, #dc2626)}.ui-form-builder__section--collapsible.mat-expansion-panel{box-shadow:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg)!important}.ui-form-builder__free-multi-field mat-chip-grid{display:flex;flex-wrap:wrap;gap:var(--ui-spacing-1)}.ui-form-builder__free-multi-field input[matInput]{flex:1;min-width:80px}.ui-form-builder mat-datepicker-toggle{display:inline-flex;align-items:center}.mdc-notched-outline__notch{border-left:none!important}.mat-expansion-panel{box-shadow:none!important}\n"] }]
|
|
801
|
+
}], propDecorators: { schema: [{
|
|
802
|
+
type: Input
|
|
803
|
+
}], initialData: [{
|
|
804
|
+
type: Input
|
|
805
|
+
}], readonly: [{
|
|
806
|
+
type: Input
|
|
807
|
+
}], disabled: [{
|
|
808
|
+
type: Input
|
|
809
|
+
}], buttonsOverride: [{
|
|
810
|
+
type: Input
|
|
811
|
+
}], loadingFor: [{
|
|
812
|
+
type: Input
|
|
813
|
+
}], valueChange: [{
|
|
814
|
+
type: Output
|
|
815
|
+
}], validationChange: [{
|
|
816
|
+
type: Output
|
|
817
|
+
}], formSubmit: [{
|
|
818
|
+
type: Output
|
|
819
|
+
}], formReset: [{
|
|
820
|
+
type: Output
|
|
821
|
+
}], customEvent: [{
|
|
822
|
+
type: Output
|
|
823
|
+
}] } });
|
|
824
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-builder.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/form-builder/form-builder.component.ts","../../../../../../packages/ng-ui-system/src/lib/components/form-builder/form-builder.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  OnInit,\r\n  OnChanges,\r\n  OnDestroy,\r\n  SimpleChanges,\r\n  AfterViewInit,\r\n  ChangeDetectorRef,\r\n  ViewEncapsulation,\r\n  inject,\r\n  ElementRef,\r\n} from '@angular/core';\r\nimport {\r\n  FormGroup,\r\n  FormControl,\r\n  FormGroupDirective,\r\n  NgForm,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n  AbstractControl,\r\n} from '@angular/forms';\r\nimport { NgTemplateOutlet } from '@angular/common';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\r\nimport { MatRadioModule } from '@angular/material/radio';\r\nimport { MatDatepickerModule } from '@angular/material/datepicker';\r\nimport {\r\n  MatNativeDateModule,\r\n  ErrorStateMatcher,\r\n  DateAdapter,\r\n  MAT_DATE_FORMATS,\r\n  MAT_DATE_LOCALE,\r\n} from '@angular/material/core';\r\nimport { MatAutocompleteModule } from '@angular/material/autocomplete';\r\nimport { MatChipsModule } from '@angular/material/chips';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subject, Subscription, Observable } from 'rxjs';\r\nimport { debounceTime, takeUntil } from 'rxjs/operators';\r\nimport { UiButtonComponent, UiButtonDescriptor } from '../../components/button/index';\r\nimport { UiButtonAreaComponent } from '../../components/button/button-area.component';\r\n\r\nimport { UiFormSchema, UiFormSection, UiFormData, UiFormCustomEvent, UiFormBuilderConfig } from './types/schema.types';\r\nimport { UiFormFieldDescriptor, UiFieldOption, UiFieldType } from './types/field.types';\r\nimport { UiFormValidationState, UiFormErrorDetail, UiValidationRule } from './types/validation.types';\r\nimport { UiFieldCondition } from './types/condition.types';\r\nimport { UiFormConditionService } from './services/form-condition.service';\r\nimport { UiFormValidationService } from './services/form-validation.service';\r\nimport { UiFormErrorSummaryComponent } from './sub-components/error-summary/form-error-summary.component';\r\nimport { UiFileInputComponent } from './sub-components/file-input/file-input.component';\r\nimport { UiSpecificaTerritorialeComponent } from './sub-components/specifica-territoriale/specifica-territoriale.component';\r\nimport { UiTableTerritorialeComponent } from './sub-components/table-territoriale/table-territoriale.component';\r\nimport { UiCurrencyInputDirective } from './directives/currency-input.directive';\r\nimport { UiItalianDateAdapter, UI_IT_DATE_FORMATS } from './adapters/it-date-adapter';\r\n\r\n/**\r\n * ErrorStateMatcher custom per il form builder.\r\n * Mostra lo stato di errore quando il campo e invalido E (touched OPPURE dirty).\r\n * Sostituisce il matcher di default di Angular Material.\r\n */\r\nexport class UiFormErrorStateMatcher implements ErrorStateMatcher {\r\n  isErrorState(control: FormControl | null, form: FormGroupDirective | NgForm | null): boolean {\r\n    return !!(control && control.invalid && (control.touched || control.dirty));\r\n  }\r\n}\r\n\r\n/**\r\n * Form builder data-driven e schema-based.\r\n *\r\n * Genera un form completo a partire da uno schema dichiarativo (`UiFormSchema`).\r\n * Supporta 16 tipi di campo, validazione condizionale, cross-field validation,\r\n * date con offset, opzioni dipendenti/dinamiche, sezioni collapsibili,\r\n * error summary con scroll-to-field e button override.\r\n *\r\n * @selector ui-form-builder\r\n *\r\n * @example\r\n * ```html\r\n * <ui-form-builder\r\n *   [schema]=\"formSchema\"\r\n *   [initialData]=\"userData\"\r\n *   (formSubmit)=\"onSubmit($event)\"\r\n *   (valueChange)=\"onChange($event)\"\r\n * />\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-form-builder',\r\n  standalone: true,\r\n  imports: [\r\n    ReactiveFormsModule,\r\n    NgTemplateOutlet,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatCheckboxModule,\r\n    MatSlideToggleModule,\r\n    MatRadioModule,\r\n    MatDatepickerModule,\r\n    MatNativeDateModule,\r\n    MatAutocompleteModule,\r\n    MatChipsModule,\r\n    MatExpansionModule,\r\n    MatTooltipModule,\r\n    LucideAngularModule,\r\n    UiButtonAreaComponent,\r\n    UiFormErrorSummaryComponent,\r\n    UiFileInputComponent,\r\n    UiSpecificaTerritorialeComponent,\r\n    UiTableTerritorialeComponent,\r\n    UiCurrencyInputDirective,\r\n  ],\r\n  providers: [\r\n    { provide: ErrorStateMatcher, useClass: UiFormErrorStateMatcher },\r\n    // Locale italiano per il datepicker Material\r\n    { provide: MAT_DATE_LOCALE, useValue: 'it-IT' },\r\n    // Adapter personalizzato per parsing DD/MM/YYYY\r\n    { provide: DateAdapter, useClass: UiItalianDateAdapter },\r\n    // Formati data italiani (DD/MM/YYYY)\r\n    { provide: MAT_DATE_FORMATS, useValue: UI_IT_DATE_FORMATS },\r\n  ],\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: { class: 'ui-form-builder-host' },\r\n  templateUrl: './form-builder.component.html',\r\n  styleUrl: './form-builder.component.scss',\r\n})\r\nexport class UiFormBuilderComponent implements OnInit, OnChanges, OnDestroy, AfterViewInit {\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n  private readonly elRef = inject(ElementRef);\r\n  private readonly conditionService = inject(UiFormConditionService);\r\n  private readonly validationService = inject(UiFormValidationService);\r\n\r\n  private readonly destroy$ = new Subject<void>();\r\n  private valueChangeSub?: Subscription;\r\n\r\n  /** ID schema corrente -- usato per evitare rebuild superflui. */\r\n  private currentSchemaId: string | null = null;\r\n\r\n  // ─── Input ──────────────────────────────────────────────────────\r\n\r\n  /** Schema del form. */\r\n  @Input() schema!: UiFormSchema;\r\n\r\n  /** Dati iniziali per la prevalorizzazione. */\r\n  @Input() initialData: UiFormData = {};\r\n\r\n  /** Modalita sola lettura globale. */\r\n  @Input() readonly = false;\r\n\r\n  /** Stato disabilitato globale. */\r\n  @Input() disabled = false;\r\n\r\n  /** Override dei pulsanti di default. */\r\n  @Input() buttonsOverride?: UiButtonDescriptor[];\r\n\r\n  /** Chiave del pulsante in stato di caricamento. */\r\n  @Input() loadingFor: string | null = null;\r\n\r\n  // ─── Output ─────────────────────────────────────────────────────\r\n\r\n  @Output() valueChange = new EventEmitter<UiFormData>();\r\n  @Output() validationChange = new EventEmitter<UiFormValidationState>();\r\n  @Output() formSubmit = new EventEmitter<UiFormData>();\r\n  @Output() formReset = new EventEmitter<void>();\r\n  @Output() customEvent = new EventEmitter<UiFormCustomEvent>();\r\n\r\n  // ─── Stato interno ──────────────────────────────────────────────\r\n\r\n  formGroup!: FormGroup;\r\n\r\n  /** Mappa campo -> opzioni filtrate (per autocomplete e dependent). */\r\n  filteredOptions: Record<string, UiFieldOption[]> = {};\r\n\r\n  /** Campi prevalorizzati (mostrano errori subito). */\r\n  prevalorizedFields = new Set<string>();\r\n\r\n  /** Mappa campo -> regole di validazione (per lookup rapido). */\r\n  private validationRulesMap: Record<string, UiValidationRule[]> = {};\r\n\r\n  /** Mappa dipendenze: campoTarget -> campi dipendenti. */\r\n  private fieldDependencies = new Map<string, Set<string>>();\r\n\r\n  /** Flag per evitare emissioni durante l'inizializzazione. */\r\n  private isInitializing = true;\r\n\r\n  // ─── Getter calcolati ───────────────────────────────────────────\r\n\r\n  get allFields(): UiFormFieldDescriptor[] {\r\n    return this.schema?.sections?.flatMap((s) => s.fields) || [];\r\n  }\r\n\r\n  get formErrors(): UiFormErrorDetail[] {\r\n    return this.getDetailedFormErrors();\r\n  }\r\n\r\n  get formButtons(): UiButtonDescriptor[] {\r\n    if (this.buttonsOverride) return this.buttonsOverride;\r\n    if (!this.schema?.config?.showButtons && this.schema?.config?.showButtons !== undefined) return [];\r\n    return [\r\n      { label: this.schema?.config?.buttonLabels?.reset || 'Reset', variant: 'outline', action: () => this.onReset() },\r\n      {\r\n        label: this.schema?.config?.buttonLabels?.submit || 'Invia',\r\n        variant: 'primary',\r\n        action: () => this.onSubmit(),\r\n      },\r\n    ];\r\n  }\r\n\r\n  // ─── Lifecycle ──────────────────────────────────────────────────\r\n  //\r\n  // ORDINE LIFECYCLE ANGULAR:\r\n  // 1. constructor\r\n  // 2. ngOnChanges (prima chiamata con valori iniziali)\r\n  // 3. ngOnInit\r\n  // 4. ngAfterViewInit\r\n  // 5. ngOnChanges (chiamate successive per cambio input)\r\n  //\r\n  // PER IL WIZARD:\r\n  // Il wizard crea un nuovo schema (con id diverso) ad ogni cambio step.\r\n  // Lo schema viene cachato nel wizard per evitare rebuild infiniti.\r\n  // ngOnChanges rileva il cambio di schema.id e ricostruisce il form.\r\n\r\n  ngOnInit(): void {\r\n    // [DEBUG] Verifica che lo schema sia disponibile al momento di ngOnInit\r\n    console.debug('[UiFormBuilder] ngOnInit - schema:', this.schema?.id, '| formGroup definito:', !!this.formGroup);\r\n    if (!this.schema) {\r\n      console.warn('[UiFormBuilder] ngOnInit - schema non presente, il form non verra costruito ora');\r\n      return;\r\n    }\r\n    this.initFormFromSchema();\r\n  }\r\n\r\n  /**\r\n   * Rileva cambiamenti sugli @Input.\r\n   * Ricostruisce il form quando cambia lo schema (id diverso).\r\n   * Cruciale per il wizard che cambia schema ad ogni step.\r\n   */\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // [DEBUG] Log di tutti i cambiamenti degli Input\r\n    const changedInputs = Object.keys(changes)\r\n      .map((k) => `${k}(firstChange=${changes[k].firstChange})`)\r\n      .join(', ');\r\n    console.debug('[UiFormBuilder] ngOnChanges -', changedInputs);\r\n\r\n    if (changes['schema'] && !changes['schema'].firstChange) {\r\n      const newSchema = changes['schema'].currentValue as UiFormSchema;\r\n      console.debug('[UiFormBuilder] ngOnChanges - nuovo schema:', newSchema?.id, '| corrente:', this.currentSchemaId);\r\n      if (newSchema && newSchema.id !== this.currentSchemaId) {\r\n        console.debug('[UiFormBuilder] ngOnChanges - REBUILD: id diverso, ricostruisco il form');\r\n        this.destroyForm();\r\n        this.initFormFromSchema();\r\n      }\r\n    }\r\n\r\n    // Aggiorna lo stato disabled/readonly globalmente\r\n    if (changes['disabled'] && !changes['disabled'].firstChange && this.formGroup) {\r\n      this.allFields.forEach((field) => {\r\n        const control = this.formGroup.get(field.key);\r\n        if (!control) return;\r\n        if (this.disabled && control.enabled) {\r\n          control.disable({ emitEvent: false });\r\n        } else if (!this.disabled && control.disabled && !field.disabled) {\r\n          control.enable({ emitEvent: false });\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    // [DEBUG] Verifica stato dopo la prima renderizzazione della vista\r\n    console.debug(\r\n      '[UiFormBuilder] ngAfterViewInit - formGroup definito:',\r\n      !!this.formGroup,\r\n      '| campi:',\r\n      this.formGroup ? Object.keys(this.formGroup.controls).length : 0,\r\n    );\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    console.debug('[UiFormBuilder] ngOnDestroy - schema:', this.currentSchemaId);\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n    this.valueChangeSub?.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Inizializza il form dallo schema corrente.\r\n   * Estratto per poter essere richiamato sia da ngOnInit che da ngOnChanges.\r\n   */\r\n  private initFormFromSchema(): void {\r\n    if (!this.schema) {\r\n      console.warn('[UiFormBuilder] initFormFromSchema - schema nullo, skip');\r\n      return;\r\n    }\r\n    console.debug(\r\n      '[UiFormBuilder] initFormFromSchema - costruisco form per:',\r\n      this.schema.id,\r\n      '| sezioni:',\r\n      this.schema.sections?.length,\r\n      '| campi totali:',\r\n      this.schema.sections?.reduce((sum, s) => sum + s.fields.length, 0),\r\n    );\r\n    this.currentSchemaId = this.schema.id;\r\n    this.buildForm();\r\n    this.isInitializing = false;\r\n    console.debug(\r\n      '[UiFormBuilder] initFormFromSchema - form costruito. Controls:',\r\n      Object.keys(this.formGroup.controls),\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Pulisce lo stato del form prima di un rebuild.\r\n   */\r\n  private destroyForm(): void {\r\n    console.debug('[UiFormBuilder] destroyForm - pulizia stato per schema:', this.currentSchemaId);\r\n    this.valueChangeSub?.unsubscribe();\r\n    this.valueChangeSub = undefined;\r\n    this.filteredOptions = {};\r\n    this.prevalorizedFields.clear();\r\n    this.validationRulesMap = {};\r\n    this.fieldDependencies.clear();\r\n    this.isInitializing = true;\r\n    this.currentSchemaId = null;\r\n  }\r\n\r\n  // ─── Costruzione form ───────────────────────────────────────────\r\n\r\n  private buildForm(): void {\r\n    const controls: Record<string, FormControl> = {};\r\n\r\n    for (const field of this.allFields) {\r\n      const initialValue = this.initialData?.[field.key] ?? field.defaultValue ?? null;\r\n      const isDisabled = field.disabled || this.disabled;\r\n\r\n      // Crea il FormControl\r\n      const control = new FormControl({ value: initialValue, disabled: isDisabled });\r\n\r\n      // Registra le regole di validazione\r\n      if (field.validation?.length) {\r\n        this.validationRulesMap[field.key] = field.validation;\r\n      }\r\n\r\n      // Traccia campi prevalorizzati\r\n      if (initialValue != null && initialValue !== '' && initialValue !== false) {\r\n        this.prevalorizedFields.add(field.key);\r\n      }\r\n\r\n      // Inizializza opzioni filtrate per select/multiselect\r\n      if (field.options && !this.isObservable(field.options)) {\r\n        this.filteredOptions[field.key] = [...(field.options as UiFieldOption[])];\r\n      }\r\n\r\n      controls[field.key] = control;\r\n    }\r\n\r\n    this.formGroup = new FormGroup(controls);\r\n\r\n    // Applica validatori iniziali\r\n    this.applyAllValidators();\r\n\r\n    // Registra dipendenze\r\n    this.buildDependencyMap();\r\n\r\n    // Sottoscrivi ai cambiamenti di valore\r\n    this.valueChangeSub = this.formGroup.valueChanges.pipe(debounceTime(50), takeUntil(this.destroy$)).subscribe(() => {\r\n      this.evaluateAllConditions();\r\n      this.updateDependentOptions();\r\n      if (!this.isInitializing) {\r\n        this.valueChange.emit(this.getFormValue());\r\n        this.validationChange.emit(this.getValidationState());\r\n        this.cdr.markForCheck();\r\n      }\r\n    });\r\n\r\n    // Valutazione condizioni iniziale\r\n    this.evaluateAllConditions();\r\n  }\r\n\r\n  // ─── Validazione ────────────────────────────────────────────────\r\n\r\n  private applyAllValidators(): void {\r\n    for (const field of this.allFields) {\r\n      if (field.validation?.length) {\r\n        const formData = this.formGroup.getRawValue();\r\n        this.validationService.updateFieldValidators(this.formGroup.get(field.key)!, field.validation, formData);\r\n      }\r\n    }\r\n  }\r\n\r\n  // ─── Condizioni ─────────────────────────────────────────────────\r\n\r\n  private evaluateAllConditions(): void {\r\n    const formData = this.formGroup.getRawValue();\r\n\r\n    for (const field of this.allFields) {\r\n      // Visibilita\r\n      if (field.conditions?.length) {\r\n        const visible = this.conditionService.evaluateConditions(field.conditions, formData);\r\n        const control = this.formGroup.get(field.key);\r\n        if (control) {\r\n          if (!visible && control.enabled) {\r\n            control.disable({ emitEvent: false });\r\n            if (!field.freezeValueOnDisable) {\r\n              control.setValue(field.defaultValue ?? null, { emitEvent: false });\r\n            }\r\n          } else if (visible && control.disabled && !field.disabled && !this.disabled) {\r\n            control.enable({ emitEvent: false });\r\n          }\r\n        }\r\n      }\r\n\r\n      // Disabilita condizionale\r\n      if (field.disableConditions?.length && !field.conditions?.length) {\r\n        const shouldDisable = this.conditionService.evaluateConditions(field.disableConditions, formData);\r\n        const control = this.formGroup.get(field.key);\r\n        if (control) {\r\n          if (shouldDisable && control.enabled) {\r\n            control.disable({ emitEvent: false });\r\n            if (!field.freezeValueOnDisable) {\r\n              control.setValue(field.defaultValue ?? null, { emitEvent: false });\r\n            }\r\n          } else if (!shouldDisable && control.disabled && !field.disabled && !this.disabled) {\r\n            control.enable({ emitEvent: false });\r\n          }\r\n        }\r\n      }\r\n\r\n      // Aggiorna validatori condizionali\r\n      if (field.validation?.some((r) => r.conditions?.length)) {\r\n        this.validationService.updateFieldValidators(this.formGroup.get(field.key)!, field.validation, formData);\r\n      }\r\n    }\r\n  }\r\n\r\n  // ─── Dipendenze opzioni ─────────────────────────────────────────\r\n\r\n  private buildDependencyMap(): void {\r\n    for (const field of this.allFields) {\r\n      if (field.dependentOptions?.dependsOn) {\r\n        const depKey = field.dependentOptions.dependsOn;\r\n        if (!this.fieldDependencies.has(depKey)) {\r\n          this.fieldDependencies.set(depKey, new Set());\r\n        }\r\n        this.fieldDependencies.get(depKey)!.add(field.key);\r\n      }\r\n    }\r\n  }\r\n\r\n  private updateDependentOptions(): void {\r\n    const formData = this.formGroup.getRawValue();\r\n\r\n    for (const field of this.allFields) {\r\n      // Dynamic options\r\n      if (field.dynamicOptions) {\r\n        if (field.dynamicOptions.type === 'year-range') {\r\n          const min = field.dynamicOptions.minYear || new Date().getFullYear() - 10;\r\n          const max = field.dynamicOptions.maxYear || new Date().getFullYear();\r\n          const opts: UiFieldOption[] = [];\r\n          for (let y = max; y >= min; y--) {\r\n            opts.push({ value: y, label: String(y) });\r\n          }\r\n          this.filteredOptions[field.key] = opts;\r\n        } else if (field.dynamicOptions.generator) {\r\n          this.filteredOptions[field.key] = field.dynamicOptions.generator(formData);\r\n        }\r\n      }\r\n\r\n      // Dependent options\r\n      if (field.dependentOptions && field.options && !this.isObservable(field.options)) {\r\n        const dep = field.dependentOptions;\r\n        const depValue = formData[dep.dependsOn];\r\n        const allOptions = field.options as UiFieldOption[];\r\n\r\n        if (depValue == null || depValue === '') {\r\n          this.filteredOptions[field.key] = [...allOptions];\r\n          continue;\r\n        }\r\n\r\n        this.filteredOptions[field.key] = allOptions.filter((opt) => {\r\n          if (dep.filterType === 'custom' && dep.filterFn) {\r\n            return dep.filterFn(opt, depValue, formData);\r\n          }\r\n          const optVal = Number(opt.value);\r\n          const depVal = Number(depValue);\r\n          if (isNaN(optVal) || isNaN(depVal)) return true;\r\n\r\n          switch (dep.filterType) {\r\n            case 'greater_than':\r\n              return optVal > depVal;\r\n            case 'greater_equal':\r\n              return optVal >= depVal;\r\n            case 'less_than':\r\n              return optVal < depVal;\r\n            case 'less_equal':\r\n              return optVal <= depVal;\r\n            case 'equals':\r\n              return optVal === depVal;\r\n            case 'not_equals':\r\n              return optVal !== depVal;\r\n            default:\r\n              return true;\r\n          }\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  // ─── API pubblica ───────────────────────────────────────────────\r\n\r\n  /** Verifica se un campo e visibile. */\r\n  isFieldVisible(fieldKey: string): boolean {\r\n    const field = this.allFields.find((f) => f.key === fieldKey);\r\n    if (!field?.conditions?.length) return true;\r\n    return this.conditionService.evaluateConditions(field.conditions, this.formGroup.getRawValue());\r\n  }\r\n\r\n  /** Verifica se un campo e obbligatorio (include validazione condizionale). */\r\n  isFieldRequired(fieldKey: string): boolean {\r\n    const field = this.allFields.find((f) => f.key === fieldKey);\r\n    if (field?.required) return true;\r\n\r\n    const rules = field?.validation || [];\r\n    const formData = this.formGroup.getRawValue();\r\n    return rules.some((r) => {\r\n      if (r.type !== 'required') return false;\r\n      if (!r.conditions?.length) return true;\r\n      return this.conditionService.evaluateConditions(r.conditions, formData);\r\n    });\r\n  }\r\n\r\n  /** Verifica se una sezione e visibile. */\r\n  isSectionVisible(sectionId: string): boolean {\r\n    const section = this.schema.sections.find((s) => s.id === sectionId);\r\n    if (!section?.conditions?.length) return true;\r\n    return this.conditionService.evaluateConditions(section.conditions, this.formGroup.getRawValue());\r\n  }\r\n\r\n  /** Verifica se mostrare errori per un campo. */\r\n  shouldShowFieldErrors(fieldKey: string): boolean {\r\n    const control = this.formGroup.get(fieldKey);\r\n    if (!control) return false;\r\n    return control.invalid && (control.dirty || control.touched || this.prevalorizedFields.has(fieldKey));\r\n  }\r\n\r\n  /** Errori di un campo come array di messaggi. */\r\n  getFieldErrors(fieldKey: string): string[] {\r\n    const control = this.formGroup.get(fieldKey);\r\n    if (!control?.errors) return [];\r\n\r\n    const field = this.allFields.find((f) => f.key === fieldKey);\r\n    const rules = field?.validation || [];\r\n    const messages: string[] = [];\r\n\r\n    for (const [errorKey] of Object.entries(control.errors)) {\r\n      // Angular usa chiavi lowercase (es. \"minlength\"), lo schema usa camelCase (es. \"minLength\").\r\n      // Normalizziamo per trovare la corrispondenza corretta.\r\n      const rule = rules.find(\r\n        (r) =>\r\n          this.normalizeValidationKey(r.type) === errorKey || (r.type === 'crossField' && errorKey === 'crossField'),\r\n      );\r\n      if (rule?.message) {\r\n        messages.push(rule.message);\r\n      } else {\r\n        messages.push(this.getDefaultErrorMessage(errorKey, control.errors![errorKey]));\r\n      }\r\n    }\r\n    return messages;\r\n  }\r\n\r\n  /** Valore completo del form (include campi disabilitati). */\r\n  getFormValue(): UiFormData {\r\n    return this.formGroup.getRawValue();\r\n  }\r\n\r\n  /** Il form e valido. */\r\n  isFormValid(): boolean {\r\n    return this.formGroup.valid;\r\n  }\r\n\r\n  /** Stato di validazione completo. */\r\n  getValidationState(): UiFormValidationState {\r\n    const errors: Record<string, string[]> = {};\r\n    const dirty: string[] = [];\r\n    const touched: string[] = [];\r\n\r\n    for (const field of this.allFields) {\r\n      const control = this.formGroup.get(field.key);\r\n      if (!control) continue;\r\n      if (control.dirty) dirty.push(field.key);\r\n      if (control.touched) touched.push(field.key);\r\n      const fieldErrors = this.getFieldErrors(field.key);\r\n      if (fieldErrors.length) errors[field.key] = fieldErrors;\r\n    }\r\n\r\n    return { valid: this.formGroup.valid, errors, dirty, touched };\r\n  }\r\n\r\n  /** Errori dettagliati per l'error summary. */\r\n  getDetailedFormErrors(): UiFormErrorDetail[] {\r\n    const details: UiFormErrorDetail[] = [];\r\n    for (const field of this.allFields) {\r\n      if (field.type === 'flag' || field.type === 'divider') continue;\r\n      if (!this.isFieldVisible(field.key)) continue;\r\n\r\n      const control = this.formGroup.get(field.key);\r\n      if (!control?.errors) continue;\r\n\r\n      const errors = this.getFieldErrors(field.key);\r\n      if (errors.length) {\r\n        details.push({ fieldKey: field.key, fieldLabel: field.label, errors });\r\n      }\r\n    }\r\n    return details;\r\n  }\r\n\r\n  /** Conta errori totali. */\r\n  getFormErrorsCount(): number {\r\n    return this.getDetailedFormErrors().reduce((sum, d) => sum + d.errors.length, 0);\r\n  }\r\n\r\n  /** Controlla se ci sono errori. */\r\n  hasFormErrors(): boolean {\r\n    return this.getFormErrorsCount() > 0;\r\n  }\r\n\r\n  /** Controllo per un campo. */\r\n  getFieldControl(key: string): FormControl | null {\r\n    return (this.formGroup.get(key) as FormControl) ?? null;\r\n  }\r\n\r\n  /** Aggiorna il valore di un campo programmaticamente. */\r\n  updateFieldValue(key: string, value: any): void {\r\n    const control = this.formGroup.get(key);\r\n    if (control) {\r\n      control.setValue(value);\r\n      control.markAsDirty();\r\n    }\r\n  }\r\n\r\n  /** Forza la validazione di tutti i campi. */\r\n  validateAllFields(): void {\r\n    Object.keys(this.formGroup.controls).forEach((key) => {\r\n      const control = this.formGroup.get(key);\r\n      control?.markAsTouched();\r\n      control?.markAsDirty();\r\n      control?.updateValueAndValidity();\r\n    });\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  /** Scrolla al campo e lo evidenzia. */\r\n  scrollToField(fieldKey: string): void {\r\n    // Forza lo stato dirty e touched per mostrare immediatamente gli errori\r\n    const control = this.formGroup.get(fieldKey);\r\n    if (control) {\r\n      control.markAsTouched();\r\n      control.markAsDirty();\r\n      control.updateValueAndValidity();\r\n    }\r\n\r\n    // Delay per lasciare che Angular Material aggiorni il DOM\r\n    setTimeout(() => {\r\n      // Cerca l'elemento tramite diversi selettori in ordine di priorita\r\n      const selectors = [`[data-field-key=\"${fieldKey}\"]`, `[formControlName=\"${fieldKey}\"]`];\r\n\r\n      let targetElement: HTMLElement | null = null;\r\n      for (const sel of selectors) {\r\n        targetElement = this.elRef.nativeElement.querySelector(sel);\r\n        if (targetElement) break;\r\n      }\r\n      if (!targetElement) return;\r\n\r\n      // Se trovato il formControlName, risali al wrapper\r\n      if (targetElement.hasAttribute('formControlName')) {\r\n        const wrapper = targetElement.closest('.ui-form-builder__field-wrapper, mat-form-field');\r\n        if (wrapper) targetElement = wrapper as HTMLElement;\r\n      }\r\n\r\n      targetElement.scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n\r\n      // Aggiungi classe highlight e rimuovila dopo l'animazione\r\n      targetElement.classList.add('ui-form-field--highlight');\r\n      setTimeout(() => targetElement!.classList.remove('ui-form-field--highlight'), 2000);\r\n    }, 100);\r\n  }\r\n\r\n  /** Submit del form. */\r\n  onSubmit(): void {\r\n    this.validateAllFields();\r\n    if (this.formGroup.valid) {\r\n      this.formSubmit.emit(this.getFormValue());\r\n    }\r\n  }\r\n\r\n  /** Reset del form. */\r\n  onReset(): void {\r\n    // Resetta a valori iniziali o default\r\n    for (const field of this.allFields) {\r\n      const value = this.initialData?.[field.key] ?? field.defaultValue ?? null;\r\n      const control = this.formGroup.get(field.key);\r\n      control?.setValue(value, { emitEvent: false });\r\n      control?.markAsPristine();\r\n      control?.markAsUntouched();\r\n    }\r\n    this.formGroup.updateValueAndValidity();\r\n    this.formReset.emit();\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // ─── Template helpers ───────────────────────────────────────────\r\n\r\n  /** Opzioni per un campo (filtrando per autocomplete query). */\r\n  getFieldOptions(field: UiFormFieldDescriptor): UiFieldOption[] {\r\n    return this.filteredOptions[field.key] || [];\r\n  }\r\n\r\n  /** Filtra opzioni per autocomplete. */\r\n  filterOptions(field: UiFormFieldDescriptor, query: string): void {\r\n    if (!query) {\r\n      if (field.options && !this.isObservable(field.options)) {\r\n        this.filteredOptions[field.key] = [...(field.options as UiFieldOption[])];\r\n      }\r\n      return;\r\n    }\r\n\r\n    const lowerQuery = query.toLowerCase();\r\n\r\n    if (field.asyncOptions) {\r\n      field.asyncOptions(query).then((opts) => {\r\n        this.filteredOptions[field.key] = opts;\r\n        this.cdr.markForCheck();\r\n      });\r\n    } else if (field.options && !this.isObservable(field.options)) {\r\n      this.filteredOptions[field.key] = (field.options as UiFieldOption[]).filter((o) =>\r\n        o.label.toLowerCase().includes(lowerQuery),\r\n      );\r\n    }\r\n  }\r\n\r\n  /** Display fn per autocomplete. */\r\n  displayFn(options: UiFieldOption[]): (value: any) => string {\r\n    return (value: any) => {\r\n      const opt = options.find((o) => o.value === value);\r\n      return opt?.label || '';\r\n    };\r\n  }\r\n\r\n  /** Seleziona tutte le opzioni per multiselect. */\r\n  selectAll(field: UiFormFieldDescriptor): void {\r\n    const options = this.getFieldOptions(field);\r\n    const allValues = options.filter((o) => !o.disabled).map((o) => o.value);\r\n    this.formGroup.get(field.key)?.setValue(allValues);\r\n  }\r\n\r\n  /** Aggiunge un chip al freemultiselect (da token: Enter o virgola). */\r\n  addFreeChip(field: UiFormFieldDescriptor, event: any): void {\r\n    const value = (event.value || '').trim();\r\n    if (!value) return;\r\n\r\n    const current = this.formGroup.get(field.key)?.value || [];\r\n    if (!current.includes(value)) {\r\n      this.formGroup.get(field.key)?.setValue([...current, value]);\r\n    }\r\n    event.chipInput?.clear();\r\n  }\r\n\r\n  /** Aggiunge un chip al freemultiselect sull'evento blur dell'input. */\r\n  addFreeChipOnBlur(field: UiFormFieldDescriptor, event: FocusEvent): void {\r\n    const input = event.target as HTMLInputElement;\r\n    const value = (input.value || '').trim();\r\n    if (!value) return;\r\n\r\n    const current = this.formGroup.get(field.key)?.value || [];\r\n    if (!current.includes(value)) {\r\n      this.formGroup.get(field.key)?.setValue([...current, value]);\r\n    }\r\n    input.value = '';\r\n  }\r\n\r\n  /** Rimuove un chip dal freemultiselect. */\r\n  removeFreeChip(field: UiFormFieldDescriptor, chipValue: string): void {\r\n    const current = this.formGroup.get(field.key)?.value || [];\r\n    const protected_ = field.metadata?.['cantDeleteList'] || [];\r\n    if (protected_.includes(chipValue)) return;\r\n    this.formGroup.get(field.key)?.setValue(current.filter((v: string) => v !== chipValue));\r\n  }\r\n\r\n  /** Verifica se un chip e protetto dalla cancellazione. */\r\n  isChipProtected(field: UiFormFieldDescriptor, chipValue: string): boolean {\r\n    return (field.metadata?.['cantDeleteList'] || []).includes(chipValue);\r\n  }\r\n\r\n  /** Rimuove chip da multiselect. */\r\n  removeMultiselectChip(field: UiFormFieldDescriptor, chipValue: any): void {\r\n    const current = this.formGroup.get(field.key)?.value || [];\r\n    this.formGroup.get(field.key)?.setValue(current.filter((v: any) => v !== chipValue));\r\n  }\r\n\r\n  /** Trova label di un'opzione dal valore. */\r\n  getOptionLabel(field: UiFormFieldDescriptor, value: any): string {\r\n    const options = this.getFieldOptions(field);\r\n    return options.find((o) => o.value === value)?.label || String(value);\r\n  }\r\n\r\n  /** Tipo di input nativo per il campo. */\r\n  getNativeInputType(field: UiFormFieldDescriptor): string {\r\n    switch (field.type) {\r\n      case 'password':\r\n        return 'password';\r\n      case 'email':\r\n        return 'email';\r\n      default:\r\n        return 'text';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Genera le classi CSS per il wrapper di un campo,\r\n   * combinando eventuali cssClasses dallo schema con le classi\r\n   * responsive per il sistema a griglia base-12.\r\n   *\r\n   * Mobile-first: tutti i campi sono span-12 di default (100%).\r\n   * Le classi `ui-col-{breakpoint}-{n}` attivano la larghezza\r\n   * configurata solo dal breakpoint indicato in su.\r\n   *\r\n   * Breakpoint supportati: sm (≥576px), md (≥768px), lg (≥1024px), xl (≥1280px).\r\n   */\r\n  getFieldWrapperClasses(field: UiFormFieldDescriptor): string {\r\n    const classes: string[] = [];\r\n\r\n    if (field.cssClasses?.length) {\r\n      classes.push(...field.cssClasses);\r\n    }\r\n\r\n    // layout.columns → classe per il breakpoint md (default desktop)\r\n    if (field.layout?.columns) {\r\n      classes.push(`ui-col-md-${field.layout.columns}`);\r\n    }\r\n\r\n    // layout.breakpoints → classi per breakpoint specifici\r\n    if (field.layout?.breakpoints) {\r\n      for (const [bp, cols] of Object.entries(field.layout.breakpoints)) {\r\n        classes.push(`ui-col-${bp}-${cols}`);\r\n      }\r\n    }\r\n\r\n    return classes.join(' ');\r\n  }\r\n\r\n  /** Character count per textarea/text con maxLength. */\r\n  getCharCount(fieldKey: string): { current: number; max: number } | null {\r\n    const field = this.allFields.find((f) => f.key === fieldKey);\r\n    const maxRule = field?.validation?.find((r) => r.type === 'maxLength');\r\n    if (!maxRule?.value) return null;\r\n\r\n    const control = this.formGroup.get(fieldKey);\r\n    return {\r\n      current: (control?.value || '').length,\r\n      max: Number(maxRule.value),\r\n    };\r\n  }\r\n\r\n  // ─── Utilita private ────────────────────────────────────────────\r\n\r\n  private isObservable(value: any): value is Observable<any> {\r\n    return value && typeof value.subscribe === 'function';\r\n  }\r\n\r\n  /**\r\n   * Normalizza la chiave del tipo di validazione dallo schema (camelCase)\r\n   * alla chiave di errore usata da Angular (lowercase).\r\n   * Es: \"minLength\" -> \"minlength\", \"maxLength\" -> \"maxlength\".\r\n   */\r\n  private normalizeValidationKey(schemaType: string): string {\r\n    const keyMap: Record<string, string> = {\r\n      minLength: 'minlength',\r\n      maxLength: 'maxlength',\r\n    };\r\n    return keyMap[schemaType] || schemaType;\r\n  }\r\n\r\n  private getDefaultErrorMessage(errorKey: string, errorValue: any): string {\r\n    // Se il valore dell'errore e gia un messaggio descrittivo (stringa),\r\n    // lo restituisce direttamente. Questo supporta i validatori personalizzati\r\n    // (es. location, location-table) che forniscono messaggi espliciti come\r\n    // valore dell'errore (es. { nazioneRequired: 'Seleziona una nazione' }).\r\n    if (typeof errorValue === 'string') {\r\n      return errorValue;\r\n    }\r\n\r\n    switch (errorKey) {\r\n      case 'required':\r\n        return 'Campo obbligatorio';\r\n      case 'email':\r\n        return 'Formato email non valido';\r\n      case 'min':\r\n        return `Valore minimo: ${errorValue.min}`;\r\n      case 'max':\r\n        return `Valore massimo: ${errorValue.max}`;\r\n      case 'minlength':\r\n        return `Lunghezza minima: ${errorValue.requiredLength} caratteri`;\r\n      case 'maxlength':\r\n        return `Lunghezza massima: ${errorValue.requiredLength} caratteri`;\r\n      case 'pattern':\r\n        return 'Formato non valido';\r\n      case 'crossField':\r\n        return 'Valore non coerente con il campo correlato';\r\n      case 'date-min':\r\n        return errorValue?.message || 'Data troppo remota';\r\n      case 'date-max':\r\n        return errorValue?.message || 'Data troppo avanzata';\r\n      case 'fileSize':\r\n        return `Dimensione massima: ${errorValue?.maxSizeFormatted}`;\r\n      case 'fileType':\r\n        return 'Formato file non accettato';\r\n      case 'fileCount':\r\n        return `Massimo ${errorValue?.maxCount} file`;\r\n      default:\r\n        return 'Valore non valido';\r\n    }\r\n  }\r\n}\r\n","<!--\r\n  ============================================================\r\n  UI FORM BUILDER - TEMPLATE PRINCIPALE\r\n  ============================================================\r\n  ARCHITETTURA:\r\n  Ogni <ng-template> che contiene direttive [formControlName]\r\n  DEVE wrappare il proprio contenuto in un <div [formGroup]>.\r\n  Questo perche Angular risolve l'injector di formControlName\r\n  dal CONTESTO DI DICHIARAZIONE del template, non dal punto\r\n  di inserimento (ngTemplateOutlet).\r\n\r\n  Senza questo wrapper, formControlName non trova il\r\n  FormGroupDirective e lancia NG01050.\r\n\r\n  Ref: versione tailored (ang-ms-webapp) usa lo stesso pattern.\r\n  ============================================================\r\n-->\r\n\r\n@if (schema && formGroup) {\r\n  <div class=\"ui-form-builder\" [formGroup]=\"formGroup\" [class]=\"schema.config?.cssClasses?.join(' ')\">\r\n\r\n    <!-- DEBUG: Verifica che formGroup sia inizializzato -->\r\n    <!-- [formGroup] e sul div padre; tutti i formControlName INLINE funzionano -->\r\n\r\n    <!-- ==================== HEADER ==================== -->\r\n    @if (schema.title || schema.description) {\r\n      <div class=\"ui-form-builder__header\">\r\n        @if (schema.title) {\r\n          <h2 class=\"ui-form-builder__title\">{{ schema.title }}</h2>\r\n        }\r\n        @if (schema.description) {\r\n          <p class=\"ui-form-builder__description\">{{ schema.description }}</p>\r\n        }\r\n      </div>\r\n    }\r\n\r\n    <!-- ==================== SEZIONI ==================== -->\r\n    <div class=\"ui-form-builder__sections\">\r\n      @for (section of schema.sections; track section.id) {\r\n        @if (isSectionVisible(section.id)) {\r\n\r\n          <!-- Sezione collassabile -->\r\n          @if (section.collapsible) {\r\n            <mat-expansion-panel\r\n              class=\"ui-form-builder__section ui-form-builder__section--collapsible\"\r\n              [class]=\"section.cssClasses?.join(' ')\"\r\n              [expanded]=\"!section.collapsed\"\r\n            >\r\n              <mat-expansion-panel-header>\r\n                @if (section.title) {\r\n                  <mat-panel-title>{{ section.title }}</mat-panel-title>\r\n                }\r\n                @if (section.description) {\r\n                  <mat-panel-description>{{ section.description }}</mat-panel-description>\r\n                }\r\n              </mat-expansion-panel-header>\r\n\r\n              <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n                @for (field of section.fields; track field.key) {\r\n                  @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n                    <div\r\n                      class=\"ui-form-builder__field-wrapper\"\r\n                      [class]=\"getFieldWrapperClasses(field)\"\r\n                      [style.order]=\"field.layout?.order || null\"\r\n                      [attr.data-field-key]=\"field.key\"\r\n                    >\r\n                      <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n                      <ng-container\r\n                        *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n                      />\r\n                    </div>\r\n                  }\r\n                }\r\n              </div>\r\n            </mat-expansion-panel>\r\n          } @else {\r\n            <!-- Sezione non collassabile -->\r\n            <div class=\"ui-form-builder__section\" [class]=\"section.cssClasses?.join(' ')\">\r\n              @if (section.title || section.description) {\r\n                <div class=\"ui-form-builder__section-header\">\r\n                  @if (section.title) {\r\n                    <h3 class=\"ui-form-builder__section-title\">{{ section.title }}</h3>\r\n                  }\r\n                  @if (section.description) {\r\n                    <p class=\"ui-form-builder__section-description\">{{ section.description }}</p>\r\n                  }\r\n                </div>\r\n              }\r\n\r\n              <div class=\"ui-form-builder__section-content\" [class.ui-form-builder__grid]=\"schema.config?.layout === 'grid' || !schema.config?.layout\">\r\n                @for (field of section.fields; track field.key) {\r\n                  @if (isFieldVisible(field.key) && field.type !== 'flag') {\r\n                    <div\r\n                      class=\"ui-form-builder__field-wrapper\"\r\n                      [class]=\"getFieldWrapperClasses(field)\"\r\n                      [style.order]=\"field.layout?.order || null\"\r\n                      [attr.data-field-key]=\"field.key\"\r\n                    >\r\n                      <!-- OUTLET: il template fieldTpl ha il proprio [formGroup] wrapper -->\r\n                      <ng-container\r\n                        *ngTemplateOutlet=\"fieldTpl; context: { $implicit: field }\"\r\n                      />\r\n                    </div>\r\n                  }\r\n                }\r\n              </div>\r\n            </div>\r\n          }\r\n        }\r\n      }\r\n    </div>\r\n\r\n    <!-- ==================== FOOTER ==================== -->\r\n    @if (!schema.config?.hideFooter) {\r\n      <div class=\"ui-form-builder__footer\">\r\n        <ui-form-error-summary\r\n          [errors]=\"formErrors\"\r\n          [totalErrorCount]=\"getFormErrorsCount()\"\r\n          (fieldClick)=\"scrollToField($event)\"\r\n        />\r\n        <ui-button-area\r\n          [buttons]=\"formButtons\"\r\n          align=\"end\"\r\n          [loadingIds]=\"loadingFor\"\r\n        />\r\n      </div>\r\n    }\r\n  </div>\r\n}\r\n\r\n<!-- ============================================================ -->\r\n<!-- FIELD TEMPLATE                                                -->\r\n<!-- ============================================================ -->\r\n<!--\r\n  CRITICO: Questo template e dichiarato FUORI dal <div [formGroup]>.\r\n  Angular risolve l'injector dal contesto di DICHIARAZIONE.\r\n  Quindi DOBBIAMO wrappare il contenuto in un <div [formGroup]>\r\n  per fornire un FormGroupDirective ai formControlName figli.\r\n  Senza questo wrapper -> errore NG01050.\r\n-->\r\n<ng-template #fieldTpl let-field>\r\n  @if (formGroup) {\r\n    <div [formGroup]=\"formGroup\" class=\"ui-fb-field-ctx\">\r\n      @switch (field.type) {\r\n\r\n        <!-- TEXT / EMAIL / PASSWORD -->\r\n        @case ('text') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n        @case ('email') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n        @case ('password') { <ng-container *ngTemplateOutlet=\"textFieldTpl; context: { $implicit: field }\" /> }\r\n\r\n        <!-- NUMBER -->\r\n        @case ('number') {\r\n          <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n            <mat-label>{{ field.label }}</mat-label>\r\n            @if (field.metadata?.type === 'currency') {\r\n              <input matInput type=\"text\" [formControlName]=\"field.key\"\r\n                [placeholder]=\"field.placeholder || ''\"\r\n                [readonly]=\"field.readonly || readonly\"\r\n                uiCurrencyInput\r\n                [currencySymbol]=\"field.metadata?.currency || 'EUR'\"\r\n                [decimalPlaces]=\"field.metadata?.decimals ?? 2\"\r\n              />\r\n            } @else {\r\n              <input matInput type=\"number\" [formControlName]=\"field.key\"\r\n                [placeholder]=\"field.placeholder || ''\"\r\n                [readonly]=\"field.readonly || readonly\"\r\n              />\r\n            }\r\n            @if (field.iconTooltip) {\r\n              <button matSuffix mat-icon-button type=\"button\"\r\n                [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n                <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n              </button>\r\n            }\r\n            <mat-error>\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <div>{{ err }}</div>\r\n              }\r\n            </mat-error>\r\n          </mat-form-field>\r\n        }\r\n\r\n        <!-- TEXTAREA -->\r\n        @case ('textarea') {\r\n          <div class=\"ui-form-builder__textarea-wrapper\">\r\n            <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n              <mat-label>{{ field.label }}</mat-label>\r\n              <textarea matInput [formControlName]=\"field.key\"\r\n                [placeholder]=\"field.placeholder || ''\"\r\n                [readonly]=\"field.readonly || readonly\"\r\n                rows=\"4\"\r\n              ></textarea>\r\n              @if (field.iconTooltip) {\r\n                <button matSuffix mat-icon-button type=\"button\"\r\n                  [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n                  <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n                </button>\r\n              }\r\n              <mat-error>\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <div>{{ err }}</div>\r\n                }\r\n              </mat-error>\r\n            </mat-form-field>\r\n            @if (getCharCount(field.key); as cc) {\r\n              <span class=\"ui-form-builder__char-count\"\r\n                [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n                [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n                {{ cc.current }} / {{ cc.max }}\r\n              </span>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- SELECT -->\r\n        @case ('select') {\r\n          @if (field.searchable) {\r\n            <!-- Autocomplete select -->\r\n            <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n              <mat-label>{{ field.label }}</mat-label>\r\n              <input matInput [formControlName]=\"field.key\"\r\n                [matAutocomplete]=\"auto\"\r\n                [placeholder]=\"field.placeholder || ''\"\r\n                [readonly]=\"field.readonly || readonly\"\r\n                (input)=\"filterOptions(field, $any($event.target).value)\"\r\n              />\r\n              <mat-autocomplete #auto=\"matAutocomplete\" [displayWith]=\"displayFn(getFieldOptions(field))\">\r\n                @for (opt of getFieldOptions(field); track opt.value) {\r\n                  <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n                    @if (opt.icon) {\r\n                      <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n                    }\r\n                    {{ opt.label }}\r\n                  </mat-option>\r\n                }\r\n              </mat-autocomplete>\r\n              <mat-error>\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <div>{{ err }}</div>\r\n                }\r\n              </mat-error>\r\n            </mat-form-field>\r\n          } @else {\r\n            <!-- Standard select -->\r\n            <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n              <mat-label>{{ field.label }}</mat-label>\r\n              <mat-select [formControlName]=\"field.key\" [placeholder]=\"field.placeholder || ''\">\r\n                @if (!field.hideEmptyOption) {\r\n                  <mat-option [value]=\"null\">--</mat-option>\r\n                }\r\n                @for (opt of getFieldOptions(field); track opt.value) {\r\n                  <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n                    @if (opt.icon) {\r\n                      <lucide-icon [name]=\"opt.icon\" [size]=\"16\" class=\"ui-form-builder__option-icon\" />\r\n                    }\r\n                    {{ opt.label }}\r\n                    @if (opt.tooltip) {\r\n                      <lucide-icon name=\"info\" [size]=\"14\"\r\n                        [matTooltip]=\"opt.tooltip\" class=\"ui-form-builder__option-info\" />\r\n                    }\r\n                  </mat-option>\r\n                }\r\n              </mat-select>\r\n              @if (field.iconTooltip) {\r\n                <button matSuffix mat-icon-button type=\"button\"\r\n                  [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n                  <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n                </button>\r\n              }\r\n              <mat-error>\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <div>{{ err }}</div>\r\n                }\r\n              </mat-error>\r\n            </mat-form-field>\r\n          }\r\n        }\r\n\r\n        <!-- MULTISELECT -->\r\n        @case ('multiselect') {\r\n          <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n            <mat-label>{{ field.label }}</mat-label>\r\n            <mat-select [formControlName]=\"field.key\" multiple [placeholder]=\"field.placeholder || ''\">\r\n              @if (field.allowSelectAll) {\r\n                <mat-option (click)=\"selectAll(field)\" (keydown.enter)=\"selectAll(field)\" (keydown.space)=\"selectAll(field)\">\r\n                  <em>Seleziona tutto</em>\r\n                </mat-option>\r\n              }\r\n              @for (opt of getFieldOptions(field); track opt.value) {\r\n                <mat-option [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n                  {{ opt.label }}\r\n                </mat-option>\r\n              }\r\n            </mat-select>\r\n            <mat-error>\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <div>{{ err }}</div>\r\n              }\r\n            </mat-error>\r\n          </mat-form-field>\r\n          <!-- Chips preview sotto il select -->\r\n          @if (formGroup.get(field.key)?.value?.length) {\r\n            <div class=\"ui-form-builder__chips-preview\">\r\n              @for (val of formGroup.get(field.key)!.value; track val) {\r\n                <span class=\"ui-form-builder__chip\">\r\n                  {{ getOptionLabel(field, val) }}\r\n                  <button type=\"button\" (click)=\"removeMultiselectChip(field, val)\"\r\n                    class=\"ui-form-builder__chip-remove\" aria-label=\"Rimuovi\">\r\n                    <lucide-icon name=\"x\" [size]=\"12\" />\r\n                  </button>\r\n                </span>\r\n              }\r\n            </div>\r\n          }\r\n        }\r\n\r\n        <!-- FREEMULTISELECT -->\r\n        @case ('freemultiselect') {\r\n          <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width ui-form-builder__free-multi-field\">\r\n            <mat-label>{{ field.label }}</mat-label>\r\n            <mat-chip-grid #chipGrid [formControlName]=\"field.key\">\r\n              @for (chip of formGroup.get(field.key)?.value || []; track chip) {\r\n                <mat-chip-row [removable]=\"!isChipProtected(field, chip)\" (removed)=\"removeFreeChip(field, chip)\">\r\n                  {{ chip }}\r\n                  @if (!isChipProtected(field, chip)) {\r\n                    <button matChipRemove aria-label=\"Rimuovi\">\r\n                      <lucide-icon name=\"x\" [size]=\"14\" />\r\n                    </button>\r\n                  }\r\n                </mat-chip-row>\r\n              }\r\n            </mat-chip-grid>\r\n            <input matInput\r\n              [matChipInputFor]=\"chipGrid\"\r\n              [placeholder]=\"field.placeholder || 'Aggiungi...'\"\r\n              [readonly]=\"field.readonly || readonly\"\r\n              (matChipInputTokenEnd)=\"addFreeChip(field, $event)\"\r\n              (blur)=\"addFreeChipOnBlur(field, $event)\"\r\n            />\r\n            <mat-error>\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <div>{{ err }}</div>\r\n              }\r\n            </mat-error>\r\n          </mat-form-field>\r\n        }\r\n\r\n        <!-- CHECKBOX -->\r\n        @case ('checkbox') {\r\n          @if (field.appearance?.style === 'switch') {\r\n            <mat-slide-toggle\r\n              [formControlName]=\"field.key\"\r\n              [color]=\"field.appearance?.color || 'primary'\"\r\n            >\r\n              {{ field.label }}\r\n            </mat-slide-toggle>\r\n          } @else {\r\n            <mat-checkbox\r\n              [formControlName]=\"field.key\"\r\n              [color]=\"field.appearance?.color || 'primary'\"\r\n            >\r\n              {{ field.label }}\r\n              @if (field.iconTooltip) {\r\n                <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n                  [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n              }\r\n            </mat-checkbox>\r\n          }\r\n          @if (shouldShowFieldErrors(field.key)) {\r\n            <div class=\"ui-form-builder__field-error\">\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <span>{{ err }}</span>\r\n              }\r\n            </div>\r\n          }\r\n        }\r\n\r\n        <!-- SWITCH -->\r\n        @case ('switch') {\r\n          <mat-slide-toggle\r\n            [formControlName]=\"field.key\"\r\n            [color]=\"field.appearance?.color || 'primary'\"\r\n          >\r\n            {{ field.label }}\r\n            @if (field.iconTooltip) {\r\n              <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"14\"\r\n                [matTooltip]=\"field.iconTooltip.text\" class=\"ui-form-builder__inline-tooltip\" />\r\n            }\r\n          </mat-slide-toggle>\r\n        }\r\n\r\n        <!-- RADIO -->\r\n        @case ('radio') {\r\n          <div class=\"ui-form-builder__radio-wrapper\">\r\n            <label class=\"ui-form-builder__field-label\">\r\n              {{ field.label }}\r\n              @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n            </label>\r\n            <mat-radio-group [formControlName]=\"field.key\" [color]=\"field.appearance?.color || 'primary'\">\r\n              @for (opt of getFieldOptions(field); track opt.value) {\r\n                <mat-radio-button [value]=\"opt.value\" [disabled]=\"opt.disabled\">\r\n                  @if (opt.icon) {\r\n                    <lucide-icon [name]=\"opt.icon\" [size]=\"16\" />\r\n                  }\r\n                  {{ opt.label }}\r\n                </mat-radio-button>\r\n              }\r\n            </mat-radio-group>\r\n            @if (shouldShowFieldErrors(field.key)) {\r\n              <div class=\"ui-form-builder__field-error\">\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <span>{{ err }}</span>\r\n                }\r\n              </div>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- DATE -->\r\n        @case ('date') {\r\n          <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n            <mat-label>{{ field.label }}</mat-label>\r\n            <input matInput [matDatepicker]=\"datepicker\" [formControlName]=\"field.key\"\r\n              [placeholder]=\"field.placeholder || 'GG/MM/AAAA'\"\r\n              [readonly]=\"field.readonly || readonly\"\r\n            />\r\n            <mat-datepicker-toggle matSuffix [for]=\"datepicker\" />\r\n            <mat-datepicker #datepicker\r\n              [startView]=\"field.customConfig?.config?.['monthView'] ? 'year' : 'month'\"\r\n            />\r\n            <mat-error>\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <div>{{ err }}</div>\r\n              }\r\n            </mat-error>\r\n          </mat-form-field>\r\n        }\r\n\r\n        <!-- DATETIME -->\r\n        @case ('datetime') {\r\n          <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n            <mat-label>{{ field.label }}</mat-label>\r\n            <input matInput type=\"datetime-local\" [formControlName]=\"field.key\"\r\n              [placeholder]=\"field.placeholder || ''\"\r\n              [readonly]=\"field.readonly || readonly\"\r\n            />\r\n            <mat-error>\r\n              @for (err of getFieldErrors(field.key); track $index) {\r\n                <div>{{ err }}</div>\r\n              }\r\n            </mat-error>\r\n          </mat-form-field>\r\n        }\r\n\r\n        <!-- FILE -->\r\n        @case ('file') {\r\n          <div class=\"ui-form-builder__file-wrapper\">\r\n            <label class=\"ui-form-builder__field-label\">\r\n              {{ field.label }}\r\n              @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n            </label>\r\n            <ui-file-input\r\n              [formControlName]=\"field.key\"\r\n              [config]=\"field.fileConfig\"\r\n            />\r\n            @if (shouldShowFieldErrors(field.key)) {\r\n              <div class=\"ui-form-builder__field-error\">\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <span>{{ err }}</span>\r\n                }\r\n              </div>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- DIVIDER -->\r\n        @case ('divider') {\r\n          <div class=\"ui-form-builder__divider\">\r\n            @if (field.label) {\r\n              <span class=\"ui-form-builder__divider-label\">{{ field.label }}</span>\r\n            }\r\n            <hr class=\"ui-form-builder__divider-line\" />\r\n          </div>\r\n        }\r\n\r\n        <!-- LOCATION (selezione territorio) -->\r\n        @case ('location') {\r\n          <div class=\"ui-form-builder__location-wrapper\">\r\n            @if (field.label) {\r\n              <label class=\"ui-form-builder__field-label\">\r\n                {{ field.label }}\r\n                @if (isFieldRequired(field.key)) { <span class=\"ui-form-builder__required\">*</span> }\r\n              </label>\r\n            }\r\n            <ui-specifica-territoriale\r\n              [formControlName]=\"field.key\"\r\n              [config]=\"field.customConfig?.config || {}\"\r\n              [disabled]=\"readonly || field.disabled || disabled\"\r\n            />\r\n            @if (shouldShowFieldErrors(field.key)) {\r\n              <div class=\"ui-form-builder__field-error\">\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <span>{{ err }}</span>\r\n                }\r\n              </div>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- LOCATION-TABLE (tabella CRUD di location) -->\r\n        @case ('location-table') {\r\n          <div class=\"ui-form-builder__location-table-wrapper\">\r\n            <ui-table-territoriale\r\n              [formControlName]=\"field.key\"\r\n              [config]=\"field.customConfig?.config || {}\"\r\n              [disabled]=\"readonly || field.disabled || disabled\"\r\n            />\r\n            @if (shouldShowFieldErrors(field.key)) {\r\n              <div class=\"ui-form-builder__field-error\">\r\n                @for (err of getFieldErrors(field.key); track $index) {\r\n                  <span>{{ err }}</span>\r\n                }\r\n              </div>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- CUSTOM -->\r\n        @case ('custom') {\r\n          <div class=\"ui-form-builder__custom-wrapper\" [attr.data-component]=\"field.customConfig?.component\">\r\n            @if (field.label) {\r\n              <label class=\"ui-form-builder__field-label\">{{ field.label }}</label>\r\n            }\r\n            <!-- I componenti custom devono essere proiettati dall'applicazione host -->\r\n            <div class=\"ui-form-builder__custom-placeholder\">\r\n              <span>Componente custom: {{ field.customConfig?.component }}</span>\r\n            </div>\r\n          </div>\r\n        }\r\n      }\r\n    </div>\r\n  }\r\n</ng-template>\r\n\r\n<!-- ============================================================ -->\r\n<!-- TEXT / EMAIL / PASSWORD template                               -->\r\n<!-- ============================================================ -->\r\n<!--\r\n  CRITICO: Anche questo template necessita del wrapper [formGroup].\r\n  Viene invocato da fieldTpl che gia ha il wrapper, ma siccome\r\n  questo e un SECONDO ng-template, Angular crea un NUOVO contesto\r\n  di dichiarazione. Il wrapper di fieldTpl non e sufficiente.\r\n-->\r\n<ng-template #textFieldTpl let-field>\r\n  @if (formGroup) {\r\n    <div [formGroup]=\"formGroup\" class=\"ui-fb-text-ctx\">\r\n      @if (field.searchable && field.type === 'text') {\r\n        <!-- Autocomplete text -->\r\n        <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n          <mat-label>{{ field.label }}</mat-label>\r\n          <input matInput [formControlName]=\"field.key\"\r\n            [type]=\"getNativeInputType(field)\"\r\n            [matAutocomplete]=\"textAuto\"\r\n            [placeholder]=\"field.placeholder || ''\"\r\n            [readonly]=\"field.readonly || readonly\"\r\n            (input)=\"filterOptions(field, $any($event.target).value)\"\r\n          />\r\n          <mat-autocomplete #textAuto=\"matAutocomplete\">\r\n            @for (opt of getFieldOptions(field); track opt.value) {\r\n              <mat-option [value]=\"opt.value\">{{ opt.label }}</mat-option>\r\n            }\r\n          </mat-autocomplete>\r\n          <mat-error>\r\n            @for (err of getFieldErrors(field.key); track $index) {\r\n              <div>{{ err }}</div>\r\n            }\r\n          </mat-error>\r\n        </mat-form-field>\r\n      } @else {\r\n        <mat-form-field appearance=\"outline\" class=\"ui-form-builder__full-width\">\r\n          <mat-label>{{ field.label }}</mat-label>\r\n          <input matInput [formControlName]=\"field.key\"\r\n            [type]=\"getNativeInputType(field)\"\r\n            [placeholder]=\"field.placeholder || ''\"\r\n            [readonly]=\"field.readonly || readonly\"\r\n          />\r\n          @if (field.iconTooltip) {\r\n            <button matSuffix mat-icon-button type=\"button\"\r\n              [matTooltip]=\"field.iconTooltip.text\" [attr.aria-label]=\"field.iconTooltip.text\">\r\n              <lucide-icon [name]=\"field.iconTooltip.icon || 'info'\" [size]=\"18\" />\r\n            </button>\r\n          }\r\n          <mat-error>\r\n            @for (err of getFieldErrors(field.key); track $index) {\r\n              <div>{{ err }}</div>\r\n            }\r\n          </mat-error>\r\n        </mat-form-field>\r\n        @if (field.type === 'text' && getCharCount(field.key); as cc) {\r\n          <span class=\"ui-form-builder__char-count\"\r\n            [class.ui-form-builder__char-count--warn]=\"cc.current > cc.max * 0.9\"\r\n            [class.ui-form-builder__char-count--error]=\"cc.current > cc.max\">\r\n            {{ cc.current }} / {{ cc.max }}\r\n          </span>\r\n        }\r\n      }\r\n    </div>\r\n  }\r\n</ng-template>\r\n"]}
|