@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.
Files changed (141) hide show
  1. package/esm2022/gnggln-ng-ui-system.mjs +5 -0
  2. package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
  3. package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
  4. package/esm2022/lib/components/accordion/index.mjs +2 -0
  5. package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
  6. package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
  7. package/esm2022/lib/components/base-layout/index.mjs +14 -0
  8. package/esm2022/lib/components/button/button-area.component.mjs +196 -0
  9. package/esm2022/lib/components/button/button.component.mjs +164 -0
  10. package/esm2022/lib/components/button/button.types.mjs +6 -0
  11. package/esm2022/lib/components/button/index.mjs +16 -0
  12. package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
  13. package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
  14. package/esm2022/lib/components/crud-table/index.mjs +16 -0
  15. package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
  16. package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
  17. package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
  18. package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
  19. package/esm2022/lib/components/form-builder/index.mjs +19 -0
  20. package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
  21. package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
  22. package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
  23. package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
  24. package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
  25. package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
  26. package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
  27. package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
  28. package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
  29. package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
  30. package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
  31. package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
  32. package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
  33. package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
  34. package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
  35. package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
  36. package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
  37. package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
  38. package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
  39. package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
  40. package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
  41. package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
  42. package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
  43. package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
  44. package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
  45. package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
  46. package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
  47. package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
  48. package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
  49. package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
  50. package/esm2022/lib/components/layout-builder/index.mjs +18 -0
  51. package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
  52. package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
  53. package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
  54. package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
  55. package/esm2022/lib/components/modal/index.mjs +4 -0
  56. package/esm2022/lib/components/modal/modal.component.mjs +139 -0
  57. package/esm2022/lib/components/modal/modal.service.mjs +194 -0
  58. package/esm2022/lib/components/modal/modal.types.mjs +6 -0
  59. package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
  60. package/esm2022/lib/components/page-header/index.mjs +20 -0
  61. package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
  62. package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
  63. package/esm2022/lib/components/table/index.mjs +2 -0
  64. package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
  65. package/esm2022/lib/components/table/table.types.mjs +6 -0
  66. package/esm2022/lib/core/types/index.mjs +6 -0
  67. package/esm2022/lib/core/utils/index.mjs +53 -0
  68. package/esm2022/lib/sources/location-data.opt.json +8942 -0
  69. package/esm2022/lib/sources/nazioni.opt.json +215 -0
  70. package/esm2022/public-api.mjs +34 -0
  71. package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
  72. package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
  73. package/index.d.ts +5 -0
  74. package/lib/components/accordion/accordion.component.d.ts +118 -0
  75. package/lib/components/accordion/accordion.types.d.ts +62 -0
  76. package/lib/components/accordion/index.d.ts +2 -0
  77. package/lib/components/base-layout/base-layout.component.d.ts +83 -0
  78. package/lib/components/base-layout/base-layout.types.d.ts +26 -0
  79. package/lib/components/base-layout/index.d.ts +13 -0
  80. package/lib/components/button/button-area.component.d.ts +88 -0
  81. package/lib/components/button/button.component.d.ts +55 -0
  82. package/lib/components/button/button.types.d.ts +70 -0
  83. package/lib/components/button/index.d.ts +15 -0
  84. package/lib/components/crud-table/crud-table.component.d.ts +143 -0
  85. package/lib/components/crud-table/crud-table.types.d.ts +207 -0
  86. package/lib/components/crud-table/index.d.ts +15 -0
  87. package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
  88. package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
  89. package/lib/components/form-builder/form-builder.component.d.ts +183 -0
  90. package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
  91. package/lib/components/form-builder/index.d.ts +13 -0
  92. package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
  93. package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
  94. package/lib/components/form-builder/services/location.service.d.ts +83 -0
  95. package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
  96. package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
  97. package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
  98. package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
  99. package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
  100. package/lib/components/form-builder/types/condition.types.d.ts +51 -0
  101. package/lib/components/form-builder/types/field.types.d.ts +288 -0
  102. package/lib/components/form-builder/types/index.d.ts +5 -0
  103. package/lib/components/form-builder/types/schema.types.d.ts +227 -0
  104. package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
  105. package/lib/components/form-builder/types/validation.types.d.ts +174 -0
  106. package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
  107. package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
  108. package/lib/components/form-builder-editor/index.d.ts +15 -0
  109. package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
  110. package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
  111. package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
  112. package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
  113. package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
  114. package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
  115. package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
  116. package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
  117. package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
  118. package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
  119. package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
  120. package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
  121. package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
  122. package/lib/components/layout-builder/index.d.ts +16 -0
  123. package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
  124. package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
  125. package/lib/components/layout-builder/layout.service.d.ts +100 -0
  126. package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
  127. package/lib/components/modal/index.d.ts +4 -0
  128. package/lib/components/modal/modal.component.d.ts +44 -0
  129. package/lib/components/modal/modal.service.d.ts +93 -0
  130. package/lib/components/modal/modal.types.d.ts +110 -0
  131. package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
  132. package/lib/components/page-header/index.d.ts +16 -0
  133. package/lib/components/page-header/page-header.component.d.ts +59 -0
  134. package/lib/components/page-header/page-header.types.d.ts +96 -0
  135. package/lib/components/table/index.d.ts +2 -0
  136. package/lib/components/table/paginated-table.component.d.ts +85 -0
  137. package/lib/components/table/table.types.d.ts +81 -0
  138. package/lib/core/types/index.d.ts +57 -0
  139. package/lib/core/utils/index.d.ts +29 -0
  140. package/package.json +44 -0
  141. 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"]}