@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,510 @@
1
+ import { Component, Input, Output, EventEmitter, ChangeDetectorRef, ViewEncapsulation, ViewChild, inject, } from '@angular/core';
2
+ import { LucideAngularModule } from 'lucide-angular';
3
+ import { Subject, takeUntil } from 'rxjs';
4
+ import { UiButtonComponent } from '../../components/button/index';
5
+ import { UiFormBuilderComponent } from './form-builder.component';
6
+ import { UiWizardSyncService } from './services/wizard-sync.service';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "lucide-angular";
9
+ /**
10
+ * Componente wizard che wrappa `UiFormBuilderComponent`
11
+ * aggiungendo navigazione multi-step, barra di progresso,
12
+ * validazione per step e sincronizzazione wireless.
13
+ *
14
+ * @selector ui-form-wizard
15
+ *
16
+ * @example
17
+ * ```html
18
+ * <ui-form-wizard
19
+ * [schema]="wizardSchema"
20
+ * [wizardConfig]="wizardConfig"
21
+ * (wizardComplete)="onComplete($event)"
22
+ * (stepChange)="onStepChange($event)"
23
+ * />
24
+ * ```
25
+ */
26
+ export class UiFormWizardComponent {
27
+ constructor() {
28
+ this.cdr = inject(ChangeDetectorRef);
29
+ this.wizardSync = inject(UiWizardSyncService);
30
+ this.destroy$ = new Subject();
31
+ /** Metodi di validazione custom per step. */
32
+ this.customValidationMethods = {};
33
+ /** Dati iniziali. */
34
+ this.initialData = {};
35
+ /** Modalita sola lettura. */
36
+ this.readonly = false;
37
+ /** Stato disabilitato. */
38
+ this.disabled = false;
39
+ /** Pulsante in caricamento. */
40
+ this.loadingFor = null;
41
+ // ─── Output ─────────────────────────────────────────────────────
42
+ this.stepChange = new EventEmitter();
43
+ this.stepComplete = new EventEmitter();
44
+ this.stepValidationFailed = new EventEmitter();
45
+ this.wizardComplete = new EventEmitter();
46
+ this.valueChange = new EventEmitter();
47
+ this.validationChange = new EventEmitter();
48
+ this.customEvent = new EventEmitter();
49
+ // ─── Stato ──────────────────────────────────────────────────────
50
+ this.state = {
51
+ currentStep: 0,
52
+ totalSteps: 0,
53
+ completedSteps: new Set(),
54
+ visitedSteps: new Set([0]),
55
+ isValid: false,
56
+ stepErrors: {},
57
+ };
58
+ /** Step ordinati per stepNumber. */
59
+ this.wizardSteps = [];
60
+ /** Dati accumulati del form. */
61
+ this.accumulatedData = {};
62
+ /**
63
+ * Schema cachato per lo step corrente.
64
+ * Evita di creare un nuovo oggetto ad ogni ciclo di change detection,
65
+ * cosa che causerebbe rebuild infiniti nel form builder.
66
+ */
67
+ this._cachedStepSchema = null;
68
+ this._cachedStepIndex = -1;
69
+ }
70
+ // ─── Getter calcolati ───────────────────────────────────────────
71
+ get currentStepSchema() {
72
+ // Se lo step non e cambiato, restituisci lo schema cachato
73
+ if (this._cachedStepIndex === this.state.currentStep && this._cachedStepSchema) {
74
+ return this._cachedStepSchema;
75
+ }
76
+ const step = this.wizardSteps[this.state.currentStep];
77
+ if (!step)
78
+ return null;
79
+ this._cachedStepSchema = {
80
+ id: this.schema.id + '-step-' + this.state.currentStep,
81
+ sections: [step],
82
+ config: { ...this.schema.config, hideFooter: true, showButtons: false },
83
+ };
84
+ this._cachedStepIndex = this.state.currentStep;
85
+ return this._cachedStepSchema;
86
+ }
87
+ get currentStepTitle() {
88
+ const step = this.wizardSteps[this.state.currentStep];
89
+ return step?.wizardStep?.stepTitle || step?.title || '';
90
+ }
91
+ get currentStepDescription() {
92
+ const step = this.wizardSteps[this.state.currentStep];
93
+ return step?.wizardStep?.stepDescription || step?.description || '';
94
+ }
95
+ get isLastStep() {
96
+ return this.state.currentStep >= this.state.totalSteps - 1;
97
+ }
98
+ get progressPercent() {
99
+ if (this.state.totalSteps <= 1)
100
+ return 100;
101
+ return Math.round((this.state.currentStep / (this.state.totalSteps - 1)) * 100);
102
+ }
103
+ // ─── Lifecycle ──────────────────────────────────────────────────
104
+ ngOnInit() {
105
+ if (!this.schema)
106
+ return;
107
+ // Estrai e ordina gli step
108
+ this.wizardSteps = this.schema.sections
109
+ .filter((s) => s.wizardStep)
110
+ .sort((a, b) => (a.wizardStep.stepNumber || 0) - (b.wizardStep.stepNumber || 0));
111
+ this.state.totalSteps = this.wizardSteps.length;
112
+ this.accumulatedData = { ...this.initialData };
113
+ // Sincronizzazione wireless
114
+ if (this.wizardConfig.wizardId) {
115
+ this.syncWizardState();
116
+ // Ascolta navigazione esterna
117
+ this.wizardSync
118
+ .getWizardState$(this.wizardConfig.wizardId)
119
+ .pipe(takeUntil(this.destroy$))
120
+ .subscribe((data) => {
121
+ if (data && data.wizardState.currentStep !== this.state.currentStep) {
122
+ this.navigateToStep(data.wizardState.currentStep);
123
+ }
124
+ });
125
+ }
126
+ }
127
+ ngOnDestroy() {
128
+ this.destroy$.next();
129
+ this.destroy$.complete();
130
+ if (this.wizardConfig?.wizardId) {
131
+ this.wizardSync.unregisterWizard(this.wizardConfig.wizardId);
132
+ }
133
+ }
134
+ // ─── Navigazione ────────────────────────────────────────────────
135
+ /** Vai allo step successivo con validazione. */
136
+ async goToNextStep() {
137
+ if (this.isLastStep)
138
+ return;
139
+ // Valida step corrente
140
+ if (!(await this.validateCurrentStep()))
141
+ return;
142
+ // Salva dati step corrente
143
+ this.state.completedSteps.add(this.state.currentStep);
144
+ await this.autoSaveIfNeeded();
145
+ this.navigateToStep(this.state.currentStep + 1);
146
+ }
147
+ /** Torna allo step precedente. */
148
+ goToPreviousStep() {
149
+ if (this.state.currentStep <= 0)
150
+ return;
151
+ this.navigateToStep(this.state.currentStep - 1);
152
+ }
153
+ /** Completa il wizard. */
154
+ async finishWizard() {
155
+ if (!(await this.validateCurrentStep()))
156
+ return;
157
+ this.state.completedSteps.add(this.state.currentStep);
158
+ await this.autoSaveIfNeeded();
159
+ this.wizardComplete.emit(this.accumulatedData);
160
+ }
161
+ /** Salva lo step corrente manualmente. */
162
+ async saveCurrentStep() {
163
+ if (this.wizardConfig.onStepSave) {
164
+ const data = this.formBuilder?.getFormValue() || {};
165
+ await this.wizardConfig.onStepSave(data, this.state.currentStep);
166
+ }
167
+ }
168
+ // ─── Callback form builder ─────────────────────────────────────
169
+ onValueChange(data) {
170
+ this.accumulatedData = { ...this.accumulatedData, ...data };
171
+ this.valueChange.emit(this.accumulatedData);
172
+ }
173
+ onValidationChange(state) {
174
+ this.state.isValid = state.valid;
175
+ this.validationChange.emit(state);
176
+ }
177
+ // ─── Navigazione interna ───────────────────────────────────────
178
+ navigateToStep(targetStep) {
179
+ if (targetStep < 0 || targetStep >= this.state.totalSteps)
180
+ return;
181
+ const from = this.state.currentStep;
182
+ this.state.currentStep = targetStep;
183
+ this.state.visitedSteps.add(targetStep);
184
+ // Invalida la cache per forzare la creazione di un nuovo schema
185
+ this._cachedStepSchema = null;
186
+ this._cachedStepIndex = -1;
187
+ this.stepChange.emit({
188
+ from,
189
+ to: targetStep,
190
+ direction: targetStep > from ? 'next' : 'previous',
191
+ });
192
+ this.syncWizardState();
193
+ this.cdr.markForCheck();
194
+ }
195
+ async validateCurrentStep() {
196
+ // Validazione form standard
197
+ this.formBuilder?.validateAllFields();
198
+ const isFormValid = this.formBuilder?.isFormValid() ?? true;
199
+ // Validazione custom per step
200
+ const step = this.wizardSteps[this.state.currentStep];
201
+ const customFnName = step?.wizardStep?.customValidationFn;
202
+ if (customFnName && this.customValidationMethods[customFnName]) {
203
+ const customValid = await this.customValidationMethods[customFnName](this.accumulatedData, this.state.currentStep, isFormValid);
204
+ if (!customValid) {
205
+ this.stepValidationFailed.emit({
206
+ step: this.state.currentStep,
207
+ errors: ['Validazione custom fallita'],
208
+ });
209
+ return false;
210
+ }
211
+ }
212
+ // Se il form non e valido, blocca la navigazione
213
+ if (!isFormValid && step?.wizardStep?.validation?.required !== false) {
214
+ const errors = this.formBuilder?.getDetailedFormErrors().flatMap((e) => e.errors) || [];
215
+ this.stepValidationFailed.emit({
216
+ step: this.state.currentStep,
217
+ errors,
218
+ });
219
+ return false;
220
+ }
221
+ this.stepComplete.emit({
222
+ step: this.state.currentStep,
223
+ data: this.formBuilder?.getFormValue() || {},
224
+ });
225
+ return true;
226
+ }
227
+ async autoSaveIfNeeded() {
228
+ if (this.wizardConfig.autoSave && this.wizardConfig.onStepSave) {
229
+ const data = this.formBuilder?.getFormValue() || {};
230
+ await this.wizardConfig.onStepSave(data, this.state.currentStep);
231
+ }
232
+ }
233
+ syncWizardState() {
234
+ if (!this.wizardConfig.wizardId)
235
+ return;
236
+ const syncData = {
237
+ wizardId: this.wizardConfig.wizardId,
238
+ wizardState: { ...this.state },
239
+ wizardSteps: this.wizardSteps.map((s) => ({
240
+ title: s.wizardStep?.stepTitle || s.title,
241
+ description: s.wizardStep?.stepDescription || s.description,
242
+ })),
243
+ isWizardMode: true,
244
+ schema: this.schema,
245
+ };
246
+ this.wizardSync.registerWizard(this.wizardConfig.wizardId, syncData);
247
+ }
248
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormWizardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
249
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiFormWizardComponent, isStandalone: true, selector: "ui-form-wizard", inputs: { schema: "schema", wizardConfig: "wizardConfig", customValidationMethods: "customValidationMethods", initialData: "initialData", readonly: "readonly", disabled: "disabled", loadingFor: "loadingFor" }, outputs: { stepChange: "stepChange", stepComplete: "stepComplete", stepValidationFailed: "stepValidationFailed", wizardComplete: "wizardComplete", valueChange: "valueChange", validationChange: "validationChange", customEvent: "customEvent" }, viewQueries: [{ propertyName: "formBuilder", first: true, predicate: UiFormBuilderComponent, descendants: true }], ngImport: i0, template: `
250
+ @if (schema && wizardConfig) {
251
+ <div class="ui-form-wizard">
252
+
253
+ <!-- Header wizard (nascosto se wireless) -->
254
+ @if (!wizardConfig.wirelessHeading) {
255
+ <div class="ui-form-wizard__header">
256
+
257
+ @if (wizardConfig.showProgress !== false) {
258
+ <!-- Barra progresso lineare -->
259
+ @if (wizardConfig.progressStyle !== 'steps') {
260
+ <div class="ui-form-wizard__progress-bar">
261
+ <div class="ui-form-wizard__progress-fill"
262
+ [style.width.%]="progressPercent"></div>
263
+ </div>
264
+ <span class="ui-form-wizard__progress-text">
265
+ {{ progressPercent }}% completato
266
+ </span>
267
+ } @else {
268
+ <!-- Step indicators -->
269
+ <div class="ui-form-wizard__steps">
270
+ @for (step of wizardSteps; track $index; let i = $index) {
271
+ <div class="ui-form-wizard__step"
272
+ [class.ui-form-wizard__step--current]="i === state.currentStep"
273
+ [class.ui-form-wizard__step--completed]="state.completedSteps.has(i)"
274
+ [class.ui-form-wizard__step--visited]="state.visitedSteps.has(i)"
275
+ >
276
+ <div class="ui-form-wizard__step-circle">
277
+ @if (state.completedSteps.has(i)) {
278
+ <lucide-icon name="check" [size]="16" />
279
+ } @else {
280
+ <span>{{ i + 1 }}</span>
281
+ }
282
+ </div>
283
+ <div class="ui-form-wizard__step-label">
284
+ {{ step.wizardStep?.stepTitle || step.title || 'Step ' + (i + 1) }}
285
+ </div>
286
+ </div>
287
+ }
288
+ </div>
289
+ }
290
+ }
291
+
292
+ <!-- Info step corrente -->
293
+ <div class="ui-form-wizard__step-info">
294
+ <h2 class="ui-form-wizard__step-title">{{ currentStepTitle }}</h2>
295
+ @if (currentStepDescription) {
296
+ <p class="ui-form-wizard__step-description">{{ currentStepDescription }}</p>
297
+ }
298
+ <span class="ui-form-wizard__step-counter">
299
+ Passo {{ state.currentStep + 1 }} di {{ state.totalSteps }}
300
+ </span>
301
+ </div>
302
+ </div>
303
+ }
304
+
305
+ <!-- Form builder per lo step corrente -->
306
+ <ui-form-builder
307
+ [schema]="currentStepSchema!"
308
+ [initialData]="initialData"
309
+ [readonly]="readonly"
310
+ [disabled]="disabled"
311
+ [loadingFor]="loadingFor"
312
+ (valueChange)="onValueChange($event)"
313
+ (validationChange)="onValidationChange($event)"
314
+ (customEvent)="customEvent.emit($event)"
315
+ />
316
+
317
+ <!-- Footer navigazione -->
318
+ @if (wizardConfig.showNavigationButtons !== false) {
319
+ <div class="ui-form-wizard__footer">
320
+ @if (wizardConfig.allowBackNavigation !== false && state.currentStep > 0) {
321
+ <ui-button
322
+ label="{{ wizardConfig.buttonLabels?.previous || 'Indietro' }}"
323
+ variant="outline"
324
+ icon="chevron-left"
325
+ (click)="goToPreviousStep()"
326
+ />
327
+ } @else {
328
+ <div></div>
329
+ }
330
+
331
+ <div class="ui-form-wizard__footer-right">
332
+ @if (wizardConfig.autoSave && wizardConfig.buttonLabels?.save) {
333
+ <ui-button
334
+ [label]="wizardConfig.buttonLabels!.save!"
335
+ variant="outline"
336
+ (click)="saveCurrentStep()"
337
+ />
338
+ }
339
+ @if (isLastStep) {
340
+ <ui-button
341
+ label="{{ wizardConfig.buttonLabels?.finish || 'Completa' }}"
342
+ variant="primary"
343
+ icon="check"
344
+ iconPosition="trailing"
345
+ (click)="finishWizard()"
346
+ />
347
+ } @else {
348
+ <ui-button
349
+ label="{{ wizardConfig.buttonLabels?.next || 'Avanti' }}"
350
+ variant="primary"
351
+ icon="chevron-right"
352
+ iconPosition="trailing"
353
+ (click)="goToNextStep()"
354
+ />
355
+ }
356
+ </div>
357
+ </div>
358
+ }
359
+ </div>
360
+ }
361
+ `, isInline: true, styles: [".ui-form-wizard{display:flex;flex-direction:column;gap:var(--ui-spacing-6);font-family:var(--ui-font-family)}.ui-form-wizard__header{display:flex;flex-direction:column;gap:var(--ui-spacing-4);padding-bottom:var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border)}.ui-form-wizard__progress-bar{height:6px;background:var(--ui-color-bg-subtle);border-radius:var(--ui-radius-full);overflow:hidden}.ui-form-wizard__progress-fill{height:100%;background:var(--ui-color-primary);border-radius:var(--ui-radius-full);transition:width var(--ui-transition-normal)}.ui-form-wizard__progress-text{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);margin-top:var(--ui-spacing-1)}.ui-form-wizard__steps{display:flex;justify-content:space-between;align-items:flex-start;gap:var(--ui-spacing-2);position:relative}.ui-form-wizard__steps:before{content:\"\";position:absolute;top:16px;left:24px;right:24px;height:2px;background:var(--ui-color-border);z-index:0}.ui-form-wizard__step{display:flex;flex-direction:column;align-items:center;gap:var(--ui-spacing-1);z-index:1;flex:1}.ui-form-wizard__step-circle{width:32px;height:32px;border-radius:var(--ui-radius-full);display:flex;align-items:center;justify-content:center;border:2px solid var(--ui-color-border);background:var(--ui-color-surface);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-muted);transition:background-color var(--ui-transition-fast),border-color var(--ui-transition-fast),color var(--ui-transition-fast)}.ui-form-wizard__step--current .ui-form-wizard__step-circle{border-color:var(--ui-color-primary);background:var(--ui-color-primary);color:#fff}.ui-form-wizard__step--completed .ui-form-wizard__step-circle{border-color:var(--ui-color-success, #16a34a);background:var(--ui-color-success, #16a34a);color:#fff}.ui-form-wizard__step--visited:not(.ui-form-wizard__step--current):not(.ui-form-wizard__step--completed) .ui-form-wizard__step-circle{border-color:var(--ui-color-primary);color:var(--ui-color-primary)}.ui-form-wizard__step-label{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);text-align:center;max-width:100px;line-height:1.3}.ui-form-wizard__step--current .ui-form-wizard__step-label{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-form-wizard__step-info{text-align:center}.ui-form-wizard__step-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-wizard__step-description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0 0 var(--ui-spacing-1)}.ui-form-wizard__step-counter{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted)}.ui-form-wizard__footer{display:flex;justify-content:space-between;align-items:center;padding-top:var(--ui-spacing-4);border-top:1px solid var(--ui-color-border)}.ui-form-wizard__footer-right{display:flex;gap:var(--ui-spacing-3)}\n"], dependencies: [{ kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }, { kind: "component", type: UiFormBuilderComponent, selector: "ui-form-builder", inputs: ["schema", "initialData", "readonly", "disabled", "buttonsOverride", "loadingFor"], outputs: ["valueChange", "validationChange", "formSubmit", "formReset", "customEvent"] }], encapsulation: i0.ViewEncapsulation.None }); }
362
+ }
363
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormWizardComponent, decorators: [{
364
+ type: Component,
365
+ args: [{ selector: 'ui-form-wizard', standalone: true, imports: [LucideAngularModule, UiButtonComponent, UiFormBuilderComponent], encapsulation: ViewEncapsulation.None, template: `
366
+ @if (schema && wizardConfig) {
367
+ <div class="ui-form-wizard">
368
+
369
+ <!-- Header wizard (nascosto se wireless) -->
370
+ @if (!wizardConfig.wirelessHeading) {
371
+ <div class="ui-form-wizard__header">
372
+
373
+ @if (wizardConfig.showProgress !== false) {
374
+ <!-- Barra progresso lineare -->
375
+ @if (wizardConfig.progressStyle !== 'steps') {
376
+ <div class="ui-form-wizard__progress-bar">
377
+ <div class="ui-form-wizard__progress-fill"
378
+ [style.width.%]="progressPercent"></div>
379
+ </div>
380
+ <span class="ui-form-wizard__progress-text">
381
+ {{ progressPercent }}% completato
382
+ </span>
383
+ } @else {
384
+ <!-- Step indicators -->
385
+ <div class="ui-form-wizard__steps">
386
+ @for (step of wizardSteps; track $index; let i = $index) {
387
+ <div class="ui-form-wizard__step"
388
+ [class.ui-form-wizard__step--current]="i === state.currentStep"
389
+ [class.ui-form-wizard__step--completed]="state.completedSteps.has(i)"
390
+ [class.ui-form-wizard__step--visited]="state.visitedSteps.has(i)"
391
+ >
392
+ <div class="ui-form-wizard__step-circle">
393
+ @if (state.completedSteps.has(i)) {
394
+ <lucide-icon name="check" [size]="16" />
395
+ } @else {
396
+ <span>{{ i + 1 }}</span>
397
+ }
398
+ </div>
399
+ <div class="ui-form-wizard__step-label">
400
+ {{ step.wizardStep?.stepTitle || step.title || 'Step ' + (i + 1) }}
401
+ </div>
402
+ </div>
403
+ }
404
+ </div>
405
+ }
406
+ }
407
+
408
+ <!-- Info step corrente -->
409
+ <div class="ui-form-wizard__step-info">
410
+ <h2 class="ui-form-wizard__step-title">{{ currentStepTitle }}</h2>
411
+ @if (currentStepDescription) {
412
+ <p class="ui-form-wizard__step-description">{{ currentStepDescription }}</p>
413
+ }
414
+ <span class="ui-form-wizard__step-counter">
415
+ Passo {{ state.currentStep + 1 }} di {{ state.totalSteps }}
416
+ </span>
417
+ </div>
418
+ </div>
419
+ }
420
+
421
+ <!-- Form builder per lo step corrente -->
422
+ <ui-form-builder
423
+ [schema]="currentStepSchema!"
424
+ [initialData]="initialData"
425
+ [readonly]="readonly"
426
+ [disabled]="disabled"
427
+ [loadingFor]="loadingFor"
428
+ (valueChange)="onValueChange($event)"
429
+ (validationChange)="onValidationChange($event)"
430
+ (customEvent)="customEvent.emit($event)"
431
+ />
432
+
433
+ <!-- Footer navigazione -->
434
+ @if (wizardConfig.showNavigationButtons !== false) {
435
+ <div class="ui-form-wizard__footer">
436
+ @if (wizardConfig.allowBackNavigation !== false && state.currentStep > 0) {
437
+ <ui-button
438
+ label="{{ wizardConfig.buttonLabels?.previous || 'Indietro' }}"
439
+ variant="outline"
440
+ icon="chevron-left"
441
+ (click)="goToPreviousStep()"
442
+ />
443
+ } @else {
444
+ <div></div>
445
+ }
446
+
447
+ <div class="ui-form-wizard__footer-right">
448
+ @if (wizardConfig.autoSave && wizardConfig.buttonLabels?.save) {
449
+ <ui-button
450
+ [label]="wizardConfig.buttonLabels!.save!"
451
+ variant="outline"
452
+ (click)="saveCurrentStep()"
453
+ />
454
+ }
455
+ @if (isLastStep) {
456
+ <ui-button
457
+ label="{{ wizardConfig.buttonLabels?.finish || 'Completa' }}"
458
+ variant="primary"
459
+ icon="check"
460
+ iconPosition="trailing"
461
+ (click)="finishWizard()"
462
+ />
463
+ } @else {
464
+ <ui-button
465
+ label="{{ wizardConfig.buttonLabels?.next || 'Avanti' }}"
466
+ variant="primary"
467
+ icon="chevron-right"
468
+ iconPosition="trailing"
469
+ (click)="goToNextStep()"
470
+ />
471
+ }
472
+ </div>
473
+ </div>
474
+ }
475
+ </div>
476
+ }
477
+ `, styles: [".ui-form-wizard{display:flex;flex-direction:column;gap:var(--ui-spacing-6);font-family:var(--ui-font-family)}.ui-form-wizard__header{display:flex;flex-direction:column;gap:var(--ui-spacing-4);padding-bottom:var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border)}.ui-form-wizard__progress-bar{height:6px;background:var(--ui-color-bg-subtle);border-radius:var(--ui-radius-full);overflow:hidden}.ui-form-wizard__progress-fill{height:100%;background:var(--ui-color-primary);border-radius:var(--ui-radius-full);transition:width var(--ui-transition-normal)}.ui-form-wizard__progress-text{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);margin-top:var(--ui-spacing-1)}.ui-form-wizard__steps{display:flex;justify-content:space-between;align-items:flex-start;gap:var(--ui-spacing-2);position:relative}.ui-form-wizard__steps:before{content:\"\";position:absolute;top:16px;left:24px;right:24px;height:2px;background:var(--ui-color-border);z-index:0}.ui-form-wizard__step{display:flex;flex-direction:column;align-items:center;gap:var(--ui-spacing-1);z-index:1;flex:1}.ui-form-wizard__step-circle{width:32px;height:32px;border-radius:var(--ui-radius-full);display:flex;align-items:center;justify-content:center;border:2px solid var(--ui-color-border);background:var(--ui-color-surface);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-muted);transition:background-color var(--ui-transition-fast),border-color var(--ui-transition-fast),color var(--ui-transition-fast)}.ui-form-wizard__step--current .ui-form-wizard__step-circle{border-color:var(--ui-color-primary);background:var(--ui-color-primary);color:#fff}.ui-form-wizard__step--completed .ui-form-wizard__step-circle{border-color:var(--ui-color-success, #16a34a);background:var(--ui-color-success, #16a34a);color:#fff}.ui-form-wizard__step--visited:not(.ui-form-wizard__step--current):not(.ui-form-wizard__step--completed) .ui-form-wizard__step-circle{border-color:var(--ui-color-primary);color:var(--ui-color-primary)}.ui-form-wizard__step-label{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);text-align:center;max-width:100px;line-height:1.3}.ui-form-wizard__step--current .ui-form-wizard__step-label{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-form-wizard__step-info{text-align:center}.ui-form-wizard__step-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-wizard__step-description{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);margin:0 0 var(--ui-spacing-1)}.ui-form-wizard__step-counter{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted)}.ui-form-wizard__footer{display:flex;justify-content:space-between;align-items:center;padding-top:var(--ui-spacing-4);border-top:1px solid var(--ui-color-border)}.ui-form-wizard__footer-right{display:flex;gap:var(--ui-spacing-3)}\n"] }]
478
+ }], propDecorators: { formBuilder: [{
479
+ type: ViewChild,
480
+ args: [UiFormBuilderComponent]
481
+ }], schema: [{
482
+ type: Input
483
+ }], wizardConfig: [{
484
+ type: Input
485
+ }], customValidationMethods: [{
486
+ type: Input
487
+ }], initialData: [{
488
+ type: Input
489
+ }], readonly: [{
490
+ type: Input
491
+ }], disabled: [{
492
+ type: Input
493
+ }], loadingFor: [{
494
+ type: Input
495
+ }], stepChange: [{
496
+ type: Output
497
+ }], stepComplete: [{
498
+ type: Output
499
+ }], stepValidationFailed: [{
500
+ type: Output
501
+ }], wizardComplete: [{
502
+ type: Output
503
+ }], valueChange: [{
504
+ type: Output
505
+ }], validationChange: [{
506
+ type: Output
507
+ }], customEvent: [{
508
+ type: Output
509
+ }] } });
510
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-wizard.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/form-builder/form-wizard.component.ts"],"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  OnDestroy,\r\n  ChangeDetectorRef,\r\n  ViewEncapsulation,\r\n  ViewChild,\r\n  inject,\r\n} from '@angular/core';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subject, takeUntil } from 'rxjs';\r\nimport { UiButtonComponent } from '../../components/button/index';\r\n\r\nimport { UiFormBuilderComponent } from './form-builder.component';\r\nimport {\r\n  UiFormSchema,\r\n  UiFormSection,\r\n  UiFormData,\r\n  UiWizardConfig,\r\n  UiWizardState,\r\n  UiWizardStepChangeEvent,\r\n  UiWizardStepCompleteEvent,\r\n  UiWizardValidationFn,\r\n  UiFormCustomEvent,\r\n} from './types/schema.types';\r\nimport { UiFormValidationState } from './types/validation.types';\r\nimport { UiWizardSyncService, UiWizardSyncData } from './services/wizard-sync.service';\r\n\r\n/**\r\n * Componente wizard che wrappa `UiFormBuilderComponent`\r\n * aggiungendo navigazione multi-step, barra di progresso,\r\n * validazione per step e sincronizzazione wireless.\r\n *\r\n * @selector ui-form-wizard\r\n *\r\n * @example\r\n * ```html\r\n * <ui-form-wizard\r\n *   [schema]=\"wizardSchema\"\r\n *   [wizardConfig]=\"wizardConfig\"\r\n *   (wizardComplete)=\"onComplete($event)\"\r\n *   (stepChange)=\"onStepChange($event)\"\r\n * />\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-form-wizard',\r\n  standalone: true,\r\n  imports: [LucideAngularModule, UiButtonComponent, UiFormBuilderComponent],\r\n  encapsulation: ViewEncapsulation.None,\r\n  template: `\r\n    @if (schema && wizardConfig) {\r\n      <div class=\"ui-form-wizard\">\r\n\r\n        <!-- Header wizard (nascosto se wireless) -->\r\n        @if (!wizardConfig.wirelessHeading) {\r\n          <div class=\"ui-form-wizard__header\">\r\n\r\n            @if (wizardConfig.showProgress !== false) {\r\n              <!-- Barra progresso lineare -->\r\n              @if (wizardConfig.progressStyle !== 'steps') {\r\n                <div class=\"ui-form-wizard__progress-bar\">\r\n                  <div class=\"ui-form-wizard__progress-fill\"\r\n                    [style.width.%]=\"progressPercent\"></div>\r\n                </div>\r\n                <span class=\"ui-form-wizard__progress-text\">\r\n                  {{ progressPercent }}% completato\r\n                </span>\r\n              } @else {\r\n                <!-- Step indicators -->\r\n                <div class=\"ui-form-wizard__steps\">\r\n                  @for (step of wizardSteps; track $index; let i = $index) {\r\n                    <div class=\"ui-form-wizard__step\"\r\n                      [class.ui-form-wizard__step--current]=\"i === state.currentStep\"\r\n                      [class.ui-form-wizard__step--completed]=\"state.completedSteps.has(i)\"\r\n                      [class.ui-form-wizard__step--visited]=\"state.visitedSteps.has(i)\"\r\n                    >\r\n                      <div class=\"ui-form-wizard__step-circle\">\r\n                        @if (state.completedSteps.has(i)) {\r\n                          <lucide-icon name=\"check\" [size]=\"16\" />\r\n                        } @else {\r\n                          <span>{{ i + 1 }}</span>\r\n                        }\r\n                      </div>\r\n                      <div class=\"ui-form-wizard__step-label\">\r\n                        {{ step.wizardStep?.stepTitle || step.title || 'Step ' + (i + 1) }}\r\n                      </div>\r\n                    </div>\r\n                  }\r\n                </div>\r\n              }\r\n            }\r\n\r\n            <!-- Info step corrente -->\r\n            <div class=\"ui-form-wizard__step-info\">\r\n              <h2 class=\"ui-form-wizard__step-title\">{{ currentStepTitle }}</h2>\r\n              @if (currentStepDescription) {\r\n                <p class=\"ui-form-wizard__step-description\">{{ currentStepDescription }}</p>\r\n              }\r\n              <span class=\"ui-form-wizard__step-counter\">\r\n                Passo {{ state.currentStep + 1 }} di {{ state.totalSteps }}\r\n              </span>\r\n            </div>\r\n          </div>\r\n        }\r\n\r\n        <!-- Form builder per lo step corrente -->\r\n        <ui-form-builder\r\n          [schema]=\"currentStepSchema!\"\r\n          [initialData]=\"initialData\"\r\n          [readonly]=\"readonly\"\r\n          [disabled]=\"disabled\"\r\n          [loadingFor]=\"loadingFor\"\r\n          (valueChange)=\"onValueChange($event)\"\r\n          (validationChange)=\"onValidationChange($event)\"\r\n          (customEvent)=\"customEvent.emit($event)\"\r\n        />\r\n\r\n        <!-- Footer navigazione -->\r\n        @if (wizardConfig.showNavigationButtons !== false) {\r\n          <div class=\"ui-form-wizard__footer\">\r\n            @if (wizardConfig.allowBackNavigation !== false && state.currentStep > 0) {\r\n              <ui-button\r\n                label=\"{{ wizardConfig.buttonLabels?.previous || 'Indietro' }}\"\r\n                variant=\"outline\"\r\n                icon=\"chevron-left\"\r\n                (click)=\"goToPreviousStep()\"\r\n              />\r\n            } @else {\r\n              <div></div>\r\n            }\r\n\r\n            <div class=\"ui-form-wizard__footer-right\">\r\n              @if (wizardConfig.autoSave && wizardConfig.buttonLabels?.save) {\r\n                <ui-button\r\n                  [label]=\"wizardConfig.buttonLabels!.save!\"\r\n                  variant=\"outline\"\r\n                  (click)=\"saveCurrentStep()\"\r\n                />\r\n              }\r\n              @if (isLastStep) {\r\n                <ui-button\r\n                  label=\"{{ wizardConfig.buttonLabels?.finish || 'Completa' }}\"\r\n                  variant=\"primary\"\r\n                  icon=\"check\"\r\n                iconPosition=\"trailing\"\r\n                (click)=\"finishWizard()\"\r\n                />\r\n              } @else {\r\n                <ui-button\r\n                  label=\"{{ wizardConfig.buttonLabels?.next || 'Avanti' }}\"\r\n                  variant=\"primary\"\r\n                  icon=\"chevron-right\"\r\n                iconPosition=\"trailing\"\r\n                (click)=\"goToNextStep()\"\r\n                />\r\n              }\r\n            </div>\r\n          </div>\r\n        }\r\n      </div>\r\n    }\r\n  `,\r\n  styleUrl: './form-wizard.component.scss',\r\n})\r\nexport class UiFormWizardComponent implements OnInit, OnDestroy {\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n  private readonly wizardSync = inject(UiWizardSyncService);\r\n  private readonly destroy$ = new Subject<void>();\r\n\r\n  @ViewChild(UiFormBuilderComponent) formBuilder?: UiFormBuilderComponent;\r\n\r\n  // ─── Input ──────────────────────────────────────────────────────\r\n\r\n  /** Schema completo del form (tutte le sezioni con wizardStep). */\r\n  @Input() schema!: UiFormSchema;\r\n\r\n  /** Configurazione del wizard. */\r\n  @Input() wizardConfig!: UiWizardConfig;\r\n\r\n  /** Metodi di validazione custom per step. */\r\n  @Input() customValidationMethods: Record<string, UiWizardValidationFn> = {};\r\n\r\n  /** Dati iniziali. */\r\n  @Input() initialData: UiFormData = {};\r\n\r\n  /** Modalita sola lettura. */\r\n  @Input() readonly = false;\r\n\r\n  /** Stato disabilitato. */\r\n  @Input() disabled = false;\r\n\r\n  /** Pulsante in caricamento. */\r\n  @Input() loadingFor: string | null = null;\r\n\r\n  // ─── Output ─────────────────────────────────────────────────────\r\n\r\n  @Output() stepChange = new EventEmitter<UiWizardStepChangeEvent>();\r\n  @Output() stepComplete = new EventEmitter<UiWizardStepCompleteEvent>();\r\n  @Output() stepValidationFailed = new EventEmitter<{ step: number; errors: string[] }>();\r\n  @Output() wizardComplete = new EventEmitter<UiFormData>();\r\n  @Output() valueChange = new EventEmitter<UiFormData>();\r\n  @Output() validationChange = new EventEmitter<UiFormValidationState>();\r\n  @Output() customEvent = new EventEmitter<UiFormCustomEvent>();\r\n\r\n  // ─── Stato ──────────────────────────────────────────────────────\r\n\r\n  state: UiWizardState = {\r\n    currentStep: 0,\r\n    totalSteps: 0,\r\n    completedSteps: new Set<number>(),\r\n    visitedSteps: new Set<number>([0]),\r\n    isValid: false,\r\n    stepErrors: {},\r\n  };\r\n\r\n  /** Step ordinati per stepNumber. */\r\n  wizardSteps: UiFormSection[] = [];\r\n\r\n  /** Dati accumulati del form. */\r\n  private accumulatedData: UiFormData = {};\r\n\r\n  /**\r\n   * Schema cachato per lo step corrente.\r\n   * Evita di creare un nuovo oggetto ad ogni ciclo di change detection,\r\n   * cosa che causerebbe rebuild infiniti nel form builder.\r\n   */\r\n  _cachedStepSchema: UiFormSchema | null = null;\r\n  private _cachedStepIndex: number = -1;\r\n\r\n  // ─── Getter calcolati ───────────────────────────────────────────\r\n\r\n  get currentStepSchema(): UiFormSchema | null {\r\n    // Se lo step non e cambiato, restituisci lo schema cachato\r\n    if (this._cachedStepIndex === this.state.currentStep && this._cachedStepSchema) {\r\n      return this._cachedStepSchema;\r\n    }\r\n\r\n    const step = this.wizardSteps[this.state.currentStep];\r\n    if (!step) return null;\r\n\r\n    this._cachedStepSchema = {\r\n      id: this.schema.id + '-step-' + this.state.currentStep,\r\n      sections: [step],\r\n      config: { ...this.schema.config, hideFooter: true, showButtons: false },\r\n    };\r\n    this._cachedStepIndex = this.state.currentStep;\r\n    return this._cachedStepSchema;\r\n  }\r\n\r\n  get currentStepTitle(): string {\r\n    const step = this.wizardSteps[this.state.currentStep];\r\n    return step?.wizardStep?.stepTitle || step?.title || '';\r\n  }\r\n\r\n  get currentStepDescription(): string {\r\n    const step = this.wizardSteps[this.state.currentStep];\r\n    return step?.wizardStep?.stepDescription || step?.description || '';\r\n  }\r\n\r\n  get isLastStep(): boolean {\r\n    return this.state.currentStep >= this.state.totalSteps - 1;\r\n  }\r\n\r\n  get progressPercent(): number {\r\n    if (this.state.totalSteps <= 1) return 100;\r\n    return Math.round((this.state.currentStep / (this.state.totalSteps - 1)) * 100);\r\n  }\r\n\r\n  // ─── Lifecycle ──────────────────────────────────────────────────\r\n\r\n  ngOnInit(): void {\r\n    if (!this.schema) return;\r\n\r\n    // Estrai e ordina gli step\r\n    this.wizardSteps = this.schema.sections\r\n      .filter((s) => s.wizardStep)\r\n      .sort((a, b) => (a.wizardStep!.stepNumber || 0) - (b.wizardStep!.stepNumber || 0));\r\n\r\n    this.state.totalSteps = this.wizardSteps.length;\r\n    this.accumulatedData = { ...this.initialData };\r\n\r\n    // Sincronizzazione wireless\r\n    if (this.wizardConfig.wizardId) {\r\n      this.syncWizardState();\r\n\r\n      // Ascolta navigazione esterna\r\n      this.wizardSync\r\n        .getWizardState$(this.wizardConfig.wizardId)\r\n        .pipe(takeUntil(this.destroy$))\r\n        .subscribe((data) => {\r\n          if (data && data.wizardState.currentStep !== this.state.currentStep) {\r\n            this.navigateToStep(data.wizardState.currentStep);\r\n          }\r\n        });\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n    if (this.wizardConfig?.wizardId) {\r\n      this.wizardSync.unregisterWizard(this.wizardConfig.wizardId);\r\n    }\r\n  }\r\n\r\n  // ─── Navigazione ────────────────────────────────────────────────\r\n\r\n  /** Vai allo step successivo con validazione. */\r\n  async goToNextStep(): Promise<void> {\r\n    if (this.isLastStep) return;\r\n\r\n    // Valida step corrente\r\n    if (!(await this.validateCurrentStep())) return;\r\n\r\n    // Salva dati step corrente\r\n    this.state.completedSteps.add(this.state.currentStep);\r\n    await this.autoSaveIfNeeded();\r\n\r\n    this.navigateToStep(this.state.currentStep + 1);\r\n  }\r\n\r\n  /** Torna allo step precedente. */\r\n  goToPreviousStep(): void {\r\n    if (this.state.currentStep <= 0) return;\r\n    this.navigateToStep(this.state.currentStep - 1);\r\n  }\r\n\r\n  /** Completa il wizard. */\r\n  async finishWizard(): Promise<void> {\r\n    if (!(await this.validateCurrentStep())) return;\r\n\r\n    this.state.completedSteps.add(this.state.currentStep);\r\n    await this.autoSaveIfNeeded();\r\n\r\n    this.wizardComplete.emit(this.accumulatedData);\r\n  }\r\n\r\n  /** Salva lo step corrente manualmente. */\r\n  async saveCurrentStep(): Promise<void> {\r\n    if (this.wizardConfig.onStepSave) {\r\n      const data = this.formBuilder?.getFormValue() || {};\r\n      await this.wizardConfig.onStepSave(data, this.state.currentStep);\r\n    }\r\n  }\r\n\r\n  // ─── Callback form builder ─────────────────────────────────────\r\n\r\n  onValueChange(data: UiFormData): void {\r\n    this.accumulatedData = { ...this.accumulatedData, ...data };\r\n    this.valueChange.emit(this.accumulatedData);\r\n  }\r\n\r\n  onValidationChange(state: UiFormValidationState): void {\r\n    this.state.isValid = state.valid;\r\n    this.validationChange.emit(state);\r\n  }\r\n\r\n  // ─── Navigazione interna ───────────────────────────────────────\r\n\r\n  private navigateToStep(targetStep: number): void {\r\n    if (targetStep < 0 || targetStep >= this.state.totalSteps) return;\r\n\r\n    const from = this.state.currentStep;\r\n    this.state.currentStep = targetStep;\r\n    this.state.visitedSteps.add(targetStep);\r\n\r\n    // Invalida la cache per forzare la creazione di un nuovo schema\r\n    this._cachedStepSchema = null;\r\n    this._cachedStepIndex = -1;\r\n\r\n    this.stepChange.emit({\r\n      from,\r\n      to: targetStep,\r\n      direction: targetStep > from ? 'next' : 'previous',\r\n    });\r\n\r\n    this.syncWizardState();\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  private async validateCurrentStep(): Promise<boolean> {\r\n    // Validazione form standard\r\n    this.formBuilder?.validateAllFields();\r\n    const isFormValid = this.formBuilder?.isFormValid() ?? true;\r\n\r\n    // Validazione custom per step\r\n    const step = this.wizardSteps[this.state.currentStep];\r\n    const customFnName = step?.wizardStep?.customValidationFn;\r\n\r\n    if (customFnName && this.customValidationMethods[customFnName]) {\r\n      const customValid = await this.customValidationMethods[customFnName](\r\n        this.accumulatedData,\r\n        this.state.currentStep,\r\n        isFormValid,\r\n      );\r\n\r\n      if (!customValid) {\r\n        this.stepValidationFailed.emit({\r\n          step: this.state.currentStep,\r\n          errors: ['Validazione custom fallita'],\r\n        });\r\n        return false;\r\n      }\r\n    }\r\n\r\n    // Se il form non e valido, blocca la navigazione\r\n    if (!isFormValid && step?.wizardStep?.validation?.required !== false) {\r\n      const errors = this.formBuilder?.getDetailedFormErrors().flatMap((e) => e.errors) || [];\r\n      this.stepValidationFailed.emit({\r\n        step: this.state.currentStep,\r\n        errors,\r\n      });\r\n      return false;\r\n    }\r\n\r\n    this.stepComplete.emit({\r\n      step: this.state.currentStep,\r\n      data: this.formBuilder?.getFormValue() || {},\r\n    });\r\n\r\n    return true;\r\n  }\r\n\r\n  private async autoSaveIfNeeded(): Promise<void> {\r\n    if (this.wizardConfig.autoSave && this.wizardConfig.onStepSave) {\r\n      const data = this.formBuilder?.getFormValue() || {};\r\n      await this.wizardConfig.onStepSave(data, this.state.currentStep);\r\n    }\r\n  }\r\n\r\n  private syncWizardState(): void {\r\n    if (!this.wizardConfig.wizardId) return;\r\n\r\n    const syncData: UiWizardSyncData = {\r\n      wizardId: this.wizardConfig.wizardId,\r\n      wizardState: { ...this.state },\r\n      wizardSteps: this.wizardSteps.map((s) => ({\r\n        title: s.wizardStep?.stepTitle || s.title,\r\n        description: s.wizardStep?.stepDescription || s.description,\r\n      })),\r\n      isWizardMode: true,\r\n      schema: this.schema,\r\n    };\r\n\r\n    this.wizardSync.registerWizard(this.wizardConfig.wizardId, syncData);\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,19 @@
1
+ // ── Componenti ──
2
+ export { UiFormBuilderComponent, UiFormErrorStateMatcher } from './form-builder.component';
3
+ export { UiFormWizardComponent } from './form-wizard.component';
4
+ export { UiFormErrorSummaryComponent } from './sub-components/error-summary/form-error-summary.component';
5
+ export { UiFileInputComponent } from './sub-components/file-input/file-input.component';
6
+ export { UiSpecificaTerritorialeComponent } from './sub-components/specifica-territoriale/specifica-territoriale.component';
7
+ export { UiTableTerritorialeComponent } from './sub-components/table-territoriale/table-territoriale.component';
8
+ // ── Servizi ──
9
+ export { UiFormConditionService } from './services/form-condition.service';
10
+ export { UiFormValidationService } from './services/form-validation.service';
11
+ export { UiLocationService, } from './services/location.service';
12
+ export { UiWizardSyncService, } from './services/wizard-sync.service';
13
+ // ── Direttive ──
14
+ export { UiCurrencyInputDirective } from './directives/currency-input.directive';
15
+ // ── Adapter e formati data ──
16
+ export { UiItalianDateAdapter, UI_IT_DATE_FORMATS } from './adapters/it-date-adapter';
17
+ // ── Tipi ──
18
+ export * from './types/index';
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Zvcm0tYnVpbGRlci9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxtQkFBbUI7QUFDbkIsT0FBTyxFQUFFLHNCQUFzQixFQUFFLHVCQUF1QixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDM0YsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDaEUsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0sNkRBQTZELENBQUM7QUFDMUcsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sa0RBQWtELENBQUM7QUFDeEYsT0FBTyxFQUFFLGdDQUFnQyxFQUFFLE1BQU0sMEVBQTBFLENBQUM7QUFDNUgsT0FBTyxFQUFFLDRCQUE0QixFQUFFLE1BQU0sa0VBQWtFLENBQUM7QUFFaEgsZ0JBQWdCO0FBQ2hCLE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBQzNFLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLG9DQUFvQyxDQUFDO0FBQzdFLE9BQU8sRUFDTCxpQkFBaUIsR0FJbEIsTUFBTSw2QkFBNkIsQ0FBQztBQUNyQyxPQUFPLEVBQ0wsbUJBQW1CLEdBRXBCLE1BQU0sZ0NBQWdDLENBQUM7QUFFeEMsa0JBQWtCO0FBQ2xCLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBRWpGLCtCQUErQjtBQUMvQixPQUFPLEVBQUUsb0JBQW9CLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUV0RixhQUFhO0FBQ2IsY0FBYyxlQUFlLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyDilIDilIAgQ29tcG9uZW50aSDilIDilIBcclxuZXhwb3J0IHsgVWlGb3JtQnVpbGRlckNvbXBvbmVudCwgVWlGb3JtRXJyb3JTdGF0ZU1hdGNoZXIgfSBmcm9tICcuL2Zvcm0tYnVpbGRlci5jb21wb25lbnQnO1xyXG5leHBvcnQgeyBVaUZvcm1XaXphcmRDb21wb25lbnQgfSBmcm9tICcuL2Zvcm0td2l6YXJkLmNvbXBvbmVudCc7XHJcbmV4cG9ydCB7IFVpRm9ybUVycm9yU3VtbWFyeUNvbXBvbmVudCB9IGZyb20gJy4vc3ViLWNvbXBvbmVudHMvZXJyb3Itc3VtbWFyeS9mb3JtLWVycm9yLXN1bW1hcnkuY29tcG9uZW50JztcclxuZXhwb3J0IHsgVWlGaWxlSW5wdXRDb21wb25lbnQgfSBmcm9tICcuL3N1Yi1jb21wb25lbnRzL2ZpbGUtaW5wdXQvZmlsZS1pbnB1dC5jb21wb25lbnQnO1xyXG5leHBvcnQgeyBVaVNwZWNpZmljYVRlcnJpdG9yaWFsZUNvbXBvbmVudCB9IGZyb20gJy4vc3ViLWNvbXBvbmVudHMvc3BlY2lmaWNhLXRlcnJpdG9yaWFsZS9zcGVjaWZpY2EtdGVycml0b3JpYWxlLmNvbXBvbmVudCc7XHJcbmV4cG9ydCB7IFVpVGFibGVUZXJyaXRvcmlhbGVDb21wb25lbnQgfSBmcm9tICcuL3N1Yi1jb21wb25lbnRzL3RhYmxlLXRlcnJpdG9yaWFsZS90YWJsZS10ZXJyaXRvcmlhbGUuY29tcG9uZW50JztcclxuXHJcbi8vIOKUgOKUgCBTZXJ2aXppIOKUgOKUgFxyXG5leHBvcnQgeyBVaUZvcm1Db25kaXRpb25TZXJ2aWNlIH0gZnJvbSAnLi9zZXJ2aWNlcy9mb3JtLWNvbmRpdGlvbi5zZXJ2aWNlJztcclxuZXhwb3J0IHsgVWlGb3JtVmFsaWRhdGlvblNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2Zvcm0tdmFsaWRhdGlvbi5zZXJ2aWNlJztcclxuZXhwb3J0IHtcclxuICBVaUxvY2F0aW9uU2VydmljZSxcclxuICB0eXBlIFVpTmF6aW9uZSxcclxuICB0eXBlIFVpUHJvdmluY2lhLFxyXG4gIHR5cGUgVWlDb211bmUsXHJcbn0gZnJvbSAnLi9zZXJ2aWNlcy9sb2NhdGlvbi5zZXJ2aWNlJztcclxuZXhwb3J0IHtcclxuICBVaVdpemFyZFN5bmNTZXJ2aWNlLFxyXG4gIHR5cGUgVWlXaXphcmRTeW5jRGF0YSxcclxufSBmcm9tICcuL3NlcnZpY2VzL3dpemFyZC1zeW5jLnNlcnZpY2UnO1xyXG5cclxuLy8g4pSA4pSAIERpcmV0dGl2ZSDilIDilIBcclxuZXhwb3J0IHsgVWlDdXJyZW5jeUlucHV0RGlyZWN0aXZlIH0gZnJvbSAnLi9kaXJlY3RpdmVzL2N1cnJlbmN5LWlucHV0LmRpcmVjdGl2ZSc7XHJcblxyXG4vLyDilIDilIAgQWRhcHRlciBlIGZvcm1hdGkgZGF0YSDilIDilIBcclxuZXhwb3J0IHsgVWlJdGFsaWFuRGF0ZUFkYXB0ZXIsIFVJX0lUX0RBVEVfRk9STUFUUyB9IGZyb20gJy4vYWRhcHRlcnMvaXQtZGF0ZS1hZGFwdGVyJztcclxuXHJcbi8vIOKUgOKUgCBUaXBpIOKUgOKUgFxyXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzL2luZGV4JztcclxuIl19