@gnggln/ng-ui-system 1.0.0-alpha.12 → 1.0.0-alpha.13

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.
@@ -10,6 +10,8 @@ import * as i1 from "lucide-angular";
10
10
  * Componente wizard che wrappa `UiFormBuilderComponent`
11
11
  * aggiungendo navigazione multi-step, barra di progresso,
12
12
  * validazione per step e sincronizzazione wireless.
13
+ * Più sezioni schema con lo stesso `wizardStep.stepNumber` formano un unico passo
14
+ * (es. più blocchi collassabili per step); metadati del passo dalla prima sezione del gruppo.
13
15
  *
14
16
  * @selector ui-form-wizard
15
17
  *
@@ -57,8 +59,11 @@ export class UiFormWizardComponent {
57
59
  isValid: false,
58
60
  stepErrors: {},
59
61
  };
60
- /** Step ordinati per stepNumber. */
61
- this.wizardSteps = [];
62
+ /**
63
+ * Gruppi di sezioni schema per passo logico del wizard.
64
+ * Sezioni con lo stesso `wizardStep.stepNumber` compongono un unico passo (ordine: come in `schema.sections` dopo sort per numero).
65
+ */
66
+ this.wizardStepGroups = [];
62
67
  /** Dati accumulati del form. */
63
68
  this.accumulatedData = {};
64
69
  /**
@@ -76,24 +81,24 @@ export class UiFormWizardComponent {
76
81
  this._cachedStepSchema) {
77
82
  return this._cachedStepSchema;
78
83
  }
79
- const step = this.wizardSteps[this.state.currentStep];
80
- if (!step)
84
+ const group = this.wizardStepGroups[this.state.currentStep];
85
+ if (!group?.length)
81
86
  return null;
82
87
  this._cachedStepSchema = {
83
88
  id: this.schema.id + '-step-' + this.state.currentStep,
84
- sections: [step],
89
+ sections: group,
85
90
  config: { ...this.schema.config, hideFooter: true, showButtons: false },
86
91
  };
87
92
  this._cachedStepIndex = this.state.currentStep;
88
93
  return this._cachedStepSchema;
89
94
  }
90
95
  get currentStepTitle() {
91
- const step = this.wizardSteps[this.state.currentStep];
92
- return step?.wizardStep?.stepTitle || step?.title || '';
96
+ const primary = this.wizardStepGroups[this.state.currentStep]?.[0];
97
+ return primary?.wizardStep?.stepTitle || primary?.title || '';
93
98
  }
94
99
  get currentStepDescription() {
95
- const step = this.wizardSteps[this.state.currentStep];
96
- return step?.wizardStep?.stepDescription || step?.description || '';
100
+ const primary = this.wizardStepGroups[this.state.currentStep]?.[0];
101
+ return primary?.wizardStep?.stepDescription || primary?.description || '';
97
102
  }
98
103
  get isLastStep() {
99
104
  return this.state.currentStep >= this.state.totalSteps - 1;
@@ -103,15 +108,21 @@ export class UiFormWizardComponent {
103
108
  return 100;
104
109
  return Math.round((this.state.currentStep / (this.state.totalSteps - 1)) * 100);
105
110
  }
111
+ /**
112
+ * Dati passati al form builder: unione tra valori forniti dal consumer e
113
+ * stato accumulato tra gli step. Senza questo, tornando indietro il builder
114
+ * riceverebbe solo `initialData` e ricreerebbe i campi vuoti, pur avendo
115
+ * i valori ancora in `accumulatedData`.
116
+ */
117
+ get mergedInitialDataForBuilder() {
118
+ return { ...this.initialData, ...this.accumulatedData };
119
+ }
106
120
  // ─── Lifecycle ──────────────────────────────────────────────────
107
121
  ngOnInit() {
108
122
  if (!this.schema)
109
123
  return;
110
- // Estrai e ordina gli step
111
- this.wizardSteps = this.schema.sections
112
- .filter((s) => s.wizardStep)
113
- .sort((a, b) => (a.wizardStep.stepNumber || 0) - (b.wizardStep.stepNumber || 0));
114
- this.state.totalSteps = this.wizardSteps.length;
124
+ this.wizardStepGroups = this.buildWizardStepGroups(this.schema.sections);
125
+ this.state.totalSteps = this.wizardStepGroups.length;
115
126
  this.accumulatedData = { ...this.initialData };
116
127
  // Sincronizzazione wireless
117
128
  if (this.wizardConfig.wizardId) {
@@ -243,14 +254,15 @@ export class UiFormWizardComponent {
243
254
  }
244
255
  async validateCurrentStep() {
245
256
  this.formBuilder?.validateAllFields();
246
- const step = this.wizardSteps[this.state.currentStep];
247
- const validateCustomComponents = step?.wizardStep?.validation?.validateCustomComponents !== false;
248
- const customFieldKeys = new Set(this.getCustomFieldKeysForStep(step));
257
+ const group = this.wizardStepGroups[this.state.currentStep];
258
+ const primary = group?.[0];
259
+ const validateCustomComponents = primary?.wizardStep?.validation?.validateCustomComponents !== false;
260
+ const customFieldKeys = new Set(this.getCustomFieldKeysForGroup(group));
249
261
  const isFormValid = validateCustomComponents
250
262
  ? (this.formBuilder?.isFormValid() ?? true)
251
263
  : (this.formBuilder?.isFormValidExcludingFieldKeys(customFieldKeys) ??
252
264
  true);
253
- const customFnName = step?.wizardStep?.customValidationFn;
265
+ const customFnName = primary?.wizardStep?.customValidationFn;
254
266
  if (customFnName && this.customValidationMethods[customFnName]) {
255
267
  const customValid = await this.customValidationMethods[customFnName](this.accumulatedData, this.state.currentStep, isFormValid);
256
268
  if (!customValid) {
@@ -261,10 +273,10 @@ export class UiFormWizardComponent {
261
273
  return false;
262
274
  }
263
275
  }
264
- if (!isFormValid && step?.wizardStep?.validation?.required !== false) {
276
+ if (!isFormValid && primary?.wizardStep?.validation?.required !== false) {
265
277
  let details = this.formBuilder?.getDetailedFormErrors() || [];
266
- if (!validateCustomComponents && step) {
267
- details = this.filterErrorDetailsExcludingCustomFields(details, step, customFieldKeys);
278
+ if (!validateCustomComponents && group?.length) {
279
+ details = this.filterErrorDetailsExcludingCustomFields(details, group, customFieldKeys);
268
280
  }
269
281
  const errors = details.flatMap((e) => e.errors);
270
282
  this.stepValidationFailed.emit({
@@ -279,19 +291,58 @@ export class UiFormWizardComponent {
279
291
  });
280
292
  return true;
281
293
  }
282
- /** Chiavi dei campi `custom` nello step (per escluderli dalla validazione wizard se richiesto). */
283
- getCustomFieldKeysForStep(step) {
284
- if (!step?.fields?.length)
294
+ /**
295
+ * Raggruppa le sezioni con `wizardStep` per `stepNumber` (ordine stabile per numero, poi ordine di apparizione nello schema).
296
+ */
297
+ buildWizardStepGroups(sections) {
298
+ const withWizard = sections.filter((s) => s.wizardStep);
299
+ withWizard.sort((a, b) => (a.wizardStep.stepNumber || 0) - (b.wizardStep.stepNumber || 0));
300
+ const groups = [];
301
+ for (const s of withWizard) {
302
+ const num = s.wizardStep.stepNumber;
303
+ const prev = groups[groups.length - 1];
304
+ if (prev?.length &&
305
+ (prev[0].wizardStep.stepNumber || 0) === (num || 0)) {
306
+ prev.push(s);
307
+ }
308
+ else {
309
+ groups.push([s]);
310
+ }
311
+ }
312
+ return groups;
313
+ }
314
+ /** Chiavi dei campi `custom` in tutte le sezioni del passo corrente. */
315
+ getCustomFieldKeysForGroup(group) {
316
+ if (!group?.length)
285
317
  return [];
286
- return step.fields.filter((f) => f.type === 'custom').map((f) => f.key);
318
+ const keys = [];
319
+ for (const section of group) {
320
+ for (const f of section.fields || []) {
321
+ if (f.type === 'custom')
322
+ keys.push(f.key);
323
+ }
324
+ }
325
+ return keys;
287
326
  }
288
327
  /** Rimuove dagli errori i campi custom quando `validateCustomComponents` è false. */
289
- filterErrorDetailsExcludingCustomFields(details, step, customKeys) {
328
+ filterErrorDetailsExcludingCustomFields(details, group, customKeys) {
329
+ const sectionIds = group.map((s) => s.id);
290
330
  return details.filter((d) => {
291
- const logicalKey = this.logicalFieldKeyFromDetailKey(d.fieldKey, step.id);
331
+ const logicalKey = this.logicalFieldKeyFromDetailKeyForGroup(d.fieldKey, sectionIds);
292
332
  return !customKeys.has(logicalKey);
293
333
  });
294
334
  }
335
+ /**
336
+ * Risolve la chiave logica del campo rispetto a tutte le sezioni del gruppo (campi flat o repeatable).
337
+ */
338
+ logicalFieldKeyFromDetailKeyForGroup(fieldKey, sectionIds) {
339
+ for (const id of sectionIds) {
340
+ const k = this.logicalFieldKeyFromDetailKey(fieldKey, id);
341
+ if (k !== fieldKey)
342
+ return k;
343
+ }
344
+ return fieldKey;
345
+ }
295
346
  /** Estrae la chiave campo da `fieldKey` flat o `sectionId[i].fieldKey`. */
296
347
  logicalFieldKeyFromDetailKey(fieldKey, sectionId) {
297
348
  const re = new RegExp(`^${this.escapeRegExp(sectionId)}\\[\\d+\\]\\.(.+)$`);
@@ -313,9 +364,9 @@ export class UiFormWizardComponent {
313
364
  const syncData = {
314
365
  wizardId: this.wizardConfig.wizardId,
315
366
  wizardState: { ...this.state },
316
- wizardSteps: this.wizardSteps.map((s) => ({
317
- title: s.wizardStep?.stepTitle || s.title,
318
- description: s.wizardStep?.stepDescription || s.description,
367
+ wizardSteps: this.wizardStepGroups.map((g) => ({
368
+ title: g[0]?.wizardStep?.stepTitle || g[0]?.title,
369
+ description: g[0]?.wizardStep?.stepDescription || g[0]?.description,
319
370
  })),
320
371
  isWizardMode: true,
321
372
  schema: this.schema,
@@ -344,7 +395,11 @@ export class UiFormWizardComponent {
344
395
  } @else {
345
396
  <!-- Step indicators -->
346
397
  <div class="ui-form-wizard__steps">
347
- @for (step of wizardSteps; track $index; let i = $index) {
398
+ @for (
399
+ group of wizardStepGroups;
400
+ track $index;
401
+ let i = $index
402
+ ) {
348
403
  <div
349
404
  class="ui-form-wizard__step"
350
405
  [class.ui-form-wizard__step--current]="
@@ -366,8 +421,8 @@ export class UiFormWizardComponent {
366
421
  </div>
367
422
  <div class="ui-form-wizard__step-label">
368
423
  {{
369
- step.wizardStep?.stepTitle ||
370
- step.title ||
424
+ group[0].wizardStep?.stepTitle ||
425
+ group[0].title ||
371
426
  'Step ' + (i + 1)
372
427
  }}
373
428
  </div>
@@ -395,7 +450,7 @@ export class UiFormWizardComponent {
395
450
  <!-- Form builder per lo step corrente -->
396
451
  <ui-form-builder
397
452
  [schema]="currentStepSchema!"
398
- [initialData]="initialData"
453
+ [initialData]="mergedInitialDataForBuilder"
399
454
  [readonly]="readonly"
400
455
  [disabled]="disabled"
401
456
  [loadingFor]="loadingFor"
@@ -483,7 +538,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
483
538
  } @else {
484
539
  <!-- Step indicators -->
485
540
  <div class="ui-form-wizard__steps">
486
- @for (step of wizardSteps; track $index; let i = $index) {
541
+ @for (
542
+ group of wizardStepGroups;
543
+ track $index;
544
+ let i = $index
545
+ ) {
487
546
  <div
488
547
  class="ui-form-wizard__step"
489
548
  [class.ui-form-wizard__step--current]="
@@ -505,8 +564,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
505
564
  </div>
506
565
  <div class="ui-form-wizard__step-label">
507
566
  {{
508
- step.wizardStep?.stepTitle ||
509
- step.title ||
567
+ group[0].wizardStep?.stepTitle ||
568
+ group[0].title ||
510
569
  'Step ' + (i + 1)
511
570
  }}
512
571
  </div>
@@ -534,7 +593,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
534
593
  <!-- Form builder per lo step corrente -->
535
594
  <ui-form-builder
536
595
  [schema]="currentStepSchema!"
537
- [initialData]="initialData"
596
+ [initialData]="mergedInitialDataForBuilder"
538
597
  [readonly]="readonly"
539
598
  [disabled]="disabled"
540
599
  [loadingFor]="loadingFor"
@@ -637,4 +696,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
637
696
  type: HostListener,
638
697
  args: ['document:keydown', ['$event']]
639
698
  }] } });
640
- //# 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  TemplateRef,\r\n  HostListener,\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 {\r\n  UiFormErrorDetail,\r\n  UiFormValidationState,\r\n} from './types/validation.types';\r\nimport {\r\n  UiWizardSyncService,\r\n  UiWizardSyncData,\r\n} 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        <!-- Header wizard (nascosto se wireless) -->\r\n        @if (!wizardConfig.wirelessHeading) {\r\n          <div class=\"ui-form-wizard__header\">\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\r\n                    class=\"ui-form-wizard__progress-fill\"\r\n                    [style.width.%]=\"progressPercent\"\r\n                  ></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\r\n                      class=\"ui-form-wizard__step\"\r\n                      [class.ui-form-wizard__step--current]=\"\r\n                        i === state.currentStep\r\n                      \"\r\n                      [class.ui-form-wizard__step--completed]=\"\r\n                        state.completedSteps.has(i)\r\n                      \"\r\n                      [class.ui-form-wizard__step--visited]=\"\r\n                        state.visitedSteps.has(i)\r\n                      \"\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                        {{\r\n                          step.wizardStep?.stepTitle ||\r\n                            step.title ||\r\n                            'Step ' + (i + 1)\r\n                        }}\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\">\r\n                  {{ currentStepDescription }}\r\n                </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        <div class=\"ui-form-wizard__content\">\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            [customFieldTemplates]=\"customFieldTemplates\"\r\n            [customFieldTemplateFallback]=\"customFieldTemplateFallback\"\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 (\r\n                wizardConfig.allowBackNavigation !== false &&\r\n                state.currentStep > 0\r\n              ) {\r\n                <ui-button\r\n                  label=\"{{\r\n                    wizardConfig.buttonLabels?.previous || 'Indietro'\r\n                  }}\"\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=\"{{\r\n                      wizardConfig.buttonLabels?.finish || 'Completa'\r\n                    }}\"\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      </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  /** Template per campi `type: 'custom'` (inoltro al form builder). */\r\n  @Input() customFieldTemplates: Record<string, TemplateRef<unknown>> = {};\r\n\r\n  /** Fallback template per campi custom (inoltro al form builder). */\r\n  @Input() customFieldTemplateFallback?: TemplateRef<unknown>;\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<{\r\n    step: number;\r\n    errors: string[];\r\n  }>();\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 (\r\n      this._cachedStepIndex === this.state.currentStep &&\r\n      this._cachedStepSchema\r\n    ) {\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(\r\n      (this.state.currentStep / (this.state.totalSteps - 1)) * 100,\r\n    );\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(\r\n        (a, b) =>\r\n          (a.wizardStep!.stepNumber || 0) - (b.wizardStep!.stepNumber || 0),\r\n      );\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  /**\r\n   * Scorciatoia: Ctrl + Freccia sinistra/destra per navigare come i pulsanti\r\n   * Indietro / Avanti (sull'ultimo step, Avanti equivale a Completa).\r\n   * Ignorata se il focus è in un campo di input per non interferire con la digitazione.\r\n   */\r\n  @HostListener('document:keydown', ['$event'])\r\n  onDocumentKeydown(event: KeyboardEvent): void {\r\n    if (!this.schema || !this.wizardConfig || this.disabled) return;\r\n    if (this.wizardConfig.showNavigationButtons === false) return;\r\n    if (!event.ctrlKey || event.altKey || event.metaKey || event.shiftKey)\r\n      return;\r\n    if (event.key !== 'ArrowLeft' && event.key !== 'ArrowRight') return;\r\n    if (this.isEditableKeyboardTarget(event.target)) return;\r\n\r\n    if (event.key === 'ArrowLeft') {\r\n      if (this.wizardConfig.allowBackNavigation === false) return;\r\n      if (this.state.currentStep <= 0) return;\r\n      event.preventDefault();\r\n      this.goToPreviousStep();\r\n      return;\r\n    }\r\n\r\n    // ArrowRight\r\n    event.preventDefault();\r\n    if (this.isLastStep) {\r\n      void this.finishWizard();\r\n    } else {\r\n      void this.goToNextStep();\r\n    }\r\n  }\r\n\r\n  /** True se il target del keydown è un elemento dove la digitazione ha priorità sulla shortcut wizard. */\r\n  private isEditableKeyboardTarget(target: EventTarget | null): boolean {\r\n    if (!target || !(target instanceof HTMLElement)) return false;\r\n    const el = target;\r\n    if (el.isContentEditable) return true;\r\n    const tag = el.tagName;\r\n    if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT') return true;\r\n    return false;\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    this.formBuilder?.validateAllFields();\r\n    const step = this.wizardSteps[this.state.currentStep];\r\n    const validateCustomComponents =\r\n      step?.wizardStep?.validation?.validateCustomComponents !== false;\r\n    const customFieldKeys = new Set(this.getCustomFieldKeysForStep(step));\r\n\r\n    const isFormValid = validateCustomComponents\r\n      ? (this.formBuilder?.isFormValid() ?? true)\r\n      : (this.formBuilder?.isFormValidExcludingFieldKeys(customFieldKeys) ??\r\n        true);\r\n\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    if (!isFormValid && step?.wizardStep?.validation?.required !== false) {\r\n      let details = this.formBuilder?.getDetailedFormErrors() || [];\r\n      if (!validateCustomComponents && step) {\r\n        details = this.filterErrorDetailsExcludingCustomFields(\r\n          details,\r\n          step,\r\n          customFieldKeys,\r\n        );\r\n      }\r\n      const errors = details.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  /** Chiavi dei campi `custom` nello step (per escluderli dalla validazione wizard se richiesto). */\r\n  private getCustomFieldKeysForStep(step: UiFormSection | undefined): string[] {\r\n    if (!step?.fields?.length) return [];\r\n    return step.fields.filter((f) => f.type === 'custom').map((f) => f.key);\r\n  }\r\n\r\n  /** Rimuove dagli errori i campi custom quando `validateCustomComponents` è false. */\r\n  private filterErrorDetailsExcludingCustomFields(\r\n    details: UiFormErrorDetail[],\r\n    step: UiFormSection,\r\n    customKeys: Set<string>,\r\n  ): UiFormErrorDetail[] {\r\n    return details.filter((d) => {\r\n      const logicalKey = this.logicalFieldKeyFromDetailKey(d.fieldKey, step.id);\r\n      return !customKeys.has(logicalKey);\r\n    });\r\n  }\r\n\r\n  /** Estrae la chiave campo da `fieldKey` flat o `sectionId[i].fieldKey`. */\r\n  private logicalFieldKeyFromDetailKey(\r\n    fieldKey: string,\r\n    sectionId: string,\r\n  ): string {\r\n    const re = new RegExp(`^${this.escapeRegExp(sectionId)}\\\\[\\\\d+\\\\]\\\\.(.+)$`);\r\n    const m = fieldKey.match(re);\r\n    return m ? m[1] : fieldKey;\r\n  }\r\n\r\n  private escapeRegExp(s: string): string {\r\n    return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\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"]}
699
+ //# 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  TemplateRef,\r\n  HostListener,\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 {\r\n  UiFormErrorDetail,\r\n  UiFormValidationState,\r\n} from './types/validation.types';\r\nimport {\r\n  UiWizardSyncService,\r\n  UiWizardSyncData,\r\n} 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 * Più sezioni schema con lo stesso `wizardStep.stepNumber` formano un unico passo\r\n * (es. più blocchi collassabili per step); metadati del passo dalla prima sezione del gruppo.\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        <!-- Header wizard (nascosto se wireless) -->\r\n        @if (!wizardConfig.wirelessHeading) {\r\n          <div class=\"ui-form-wizard__header\">\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\r\n                    class=\"ui-form-wizard__progress-fill\"\r\n                    [style.width.%]=\"progressPercent\"\r\n                  ></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 (\r\n                    group of wizardStepGroups;\r\n                    track $index;\r\n                    let i = $index\r\n                  ) {\r\n                    <div\r\n                      class=\"ui-form-wizard__step\"\r\n                      [class.ui-form-wizard__step--current]=\"\r\n                        i === state.currentStep\r\n                      \"\r\n                      [class.ui-form-wizard__step--completed]=\"\r\n                        state.completedSteps.has(i)\r\n                      \"\r\n                      [class.ui-form-wizard__step--visited]=\"\r\n                        state.visitedSteps.has(i)\r\n                      \"\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                        {{\r\n                          group[0].wizardStep?.stepTitle ||\r\n                            group[0].title ||\r\n                            'Step ' + (i + 1)\r\n                        }}\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\">\r\n                  {{ currentStepDescription }}\r\n                </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        <div class=\"ui-form-wizard__content\">\r\n          <!-- Form builder per lo step corrente -->\r\n          <ui-form-builder\r\n            [schema]=\"currentStepSchema!\"\r\n            [initialData]=\"mergedInitialDataForBuilder\"\r\n            [readonly]=\"readonly\"\r\n            [disabled]=\"disabled\"\r\n            [loadingFor]=\"loadingFor\"\r\n            [customFieldTemplates]=\"customFieldTemplates\"\r\n            [customFieldTemplateFallback]=\"customFieldTemplateFallback\"\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 (\r\n                wizardConfig.allowBackNavigation !== false &&\r\n                state.currentStep > 0\r\n              ) {\r\n                <ui-button\r\n                  label=\"{{\r\n                    wizardConfig.buttonLabels?.previous || 'Indietro'\r\n                  }}\"\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=\"{{\r\n                      wizardConfig.buttonLabels?.finish || 'Completa'\r\n                    }}\"\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      </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  /** Template per campi `type: 'custom'` (inoltro al form builder). */\r\n  @Input() customFieldTemplates: Record<string, TemplateRef<unknown>> = {};\r\n\r\n  /** Fallback template per campi custom (inoltro al form builder). */\r\n  @Input() customFieldTemplateFallback?: TemplateRef<unknown>;\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<{\r\n    step: number;\r\n    errors: string[];\r\n  }>();\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  /**\r\n   * Gruppi di sezioni schema per passo logico del wizard.\r\n   * Sezioni con lo stesso `wizardStep.stepNumber` compongono un unico passo (ordine: come in `schema.sections` dopo sort per numero).\r\n   */\r\n  wizardStepGroups: 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 (\r\n      this._cachedStepIndex === this.state.currentStep &&\r\n      this._cachedStepSchema\r\n    ) {\r\n      return this._cachedStepSchema;\r\n    }\r\n\r\n    const group = this.wizardStepGroups[this.state.currentStep];\r\n    if (!group?.length) return null;\r\n\r\n    this._cachedStepSchema = {\r\n      id: this.schema.id + '-step-' + this.state.currentStep,\r\n      sections: group,\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 primary = this.wizardStepGroups[this.state.currentStep]?.[0];\r\n    return primary?.wizardStep?.stepTitle || primary?.title || '';\r\n  }\r\n\r\n  get currentStepDescription(): string {\r\n    const primary = this.wizardStepGroups[this.state.currentStep]?.[0];\r\n    return primary?.wizardStep?.stepDescription || primary?.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(\r\n      (this.state.currentStep / (this.state.totalSteps - 1)) * 100,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Dati passati al form builder: unione tra valori forniti dal consumer e\r\n   * stato accumulato tra gli step. Senza questo, tornando indietro il builder\r\n   * riceverebbe solo `initialData` e ricreerebbe i campi vuoti, pur avendo\r\n   * i valori ancora in `accumulatedData`.\r\n   */\r\n  get mergedInitialDataForBuilder(): UiFormData {\r\n    return { ...this.initialData, ...this.accumulatedData };\r\n  }\r\n\r\n  // ─── Lifecycle ──────────────────────────────────────────────────\r\n\r\n  ngOnInit(): void {\r\n    if (!this.schema) return;\r\n\r\n    this.wizardStepGroups = this.buildWizardStepGroups(this.schema.sections);\r\n    this.state.totalSteps = this.wizardStepGroups.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  /**\r\n   * Scorciatoia: Ctrl + Freccia sinistra/destra per navigare come i pulsanti\r\n   * Indietro / Avanti (sull'ultimo step, Avanti equivale a Completa).\r\n   * Ignorata se il focus è in un campo di input per non interferire con la digitazione.\r\n   */\r\n  @HostListener('document:keydown', ['$event'])\r\n  onDocumentKeydown(event: KeyboardEvent): void {\r\n    if (!this.schema || !this.wizardConfig || this.disabled) return;\r\n    if (this.wizardConfig.showNavigationButtons === false) return;\r\n    if (!event.ctrlKey || event.altKey || event.metaKey || event.shiftKey)\r\n      return;\r\n    if (event.key !== 'ArrowLeft' && event.key !== 'ArrowRight') return;\r\n    if (this.isEditableKeyboardTarget(event.target)) return;\r\n\r\n    if (event.key === 'ArrowLeft') {\r\n      if (this.wizardConfig.allowBackNavigation === false) return;\r\n      if (this.state.currentStep <= 0) return;\r\n      event.preventDefault();\r\n      this.goToPreviousStep();\r\n      return;\r\n    }\r\n\r\n    // ArrowRight\r\n    event.preventDefault();\r\n    if (this.isLastStep) {\r\n      void this.finishWizard();\r\n    } else {\r\n      void this.goToNextStep();\r\n    }\r\n  }\r\n\r\n  /** True se il target del keydown è un elemento dove la digitazione ha priorità sulla shortcut wizard. */\r\n  private isEditableKeyboardTarget(target: EventTarget | null): boolean {\r\n    if (!target || !(target instanceof HTMLElement)) return false;\r\n    const el = target;\r\n    if (el.isContentEditable) return true;\r\n    const tag = el.tagName;\r\n    if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT') return true;\r\n    return false;\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    this.formBuilder?.validateAllFields();\r\n    const group = this.wizardStepGroups[this.state.currentStep];\r\n    const primary = group?.[0];\r\n    const validateCustomComponents =\r\n      primary?.wizardStep?.validation?.validateCustomComponents !== false;\r\n    const customFieldKeys = new Set(this.getCustomFieldKeysForGroup(group));\r\n\r\n    const isFormValid = validateCustomComponents\r\n      ? (this.formBuilder?.isFormValid() ?? true)\r\n      : (this.formBuilder?.isFormValidExcludingFieldKeys(customFieldKeys) ??\r\n        true);\r\n\r\n    const customFnName = primary?.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    if (!isFormValid && primary?.wizardStep?.validation?.required !== false) {\r\n      let details = this.formBuilder?.getDetailedFormErrors() || [];\r\n      if (!validateCustomComponents && group?.length) {\r\n        details = this.filterErrorDetailsExcludingCustomFields(\r\n          details,\r\n          group,\r\n          customFieldKeys,\r\n        );\r\n      }\r\n      const errors = details.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  /**\r\n   * Raggruppa le sezioni con `wizardStep` per `stepNumber` (ordine stabile per numero, poi ordine di apparizione nello schema).\r\n   */\r\n  private buildWizardStepGroups(sections: UiFormSection[]): UiFormSection[][] {\r\n    const withWizard = sections.filter((s) => s.wizardStep);\r\n    withWizard.sort(\r\n      (a, b) =>\r\n        (a.wizardStep!.stepNumber || 0) - (b.wizardStep!.stepNumber || 0),\r\n    );\r\n    const groups: UiFormSection[][] = [];\r\n    for (const s of withWizard) {\r\n      const num = s.wizardStep!.stepNumber;\r\n      const prev = groups[groups.length - 1];\r\n      if (\r\n        prev?.length &&\r\n        (prev[0].wizardStep!.stepNumber || 0) === (num || 0)\r\n      ) {\r\n        prev.push(s);\r\n      } else {\r\n        groups.push([s]);\r\n      }\r\n    }\r\n    return groups;\r\n  }\r\n\r\n  /** Chiavi dei campi `custom` in tutte le sezioni del passo corrente. */\r\n  private getCustomFieldKeysForGroup(\r\n    group: UiFormSection[] | undefined,\r\n  ): string[] {\r\n    if (!group?.length) return [];\r\n    const keys: string[] = [];\r\n    for (const section of group) {\r\n      for (const f of section.fields || []) {\r\n        if (f.type === 'custom') keys.push(f.key);\r\n      }\r\n    }\r\n    return keys;\r\n  }\r\n\r\n  /** Rimuove dagli errori i campi custom quando `validateCustomComponents` è false. */\r\n  private filterErrorDetailsExcludingCustomFields(\r\n    details: UiFormErrorDetail[],\r\n    group: UiFormSection[],\r\n    customKeys: Set<string>,\r\n  ): UiFormErrorDetail[] {\r\n    const sectionIds = group.map((s) => s.id);\r\n    return details.filter((d) => {\r\n      const logicalKey = this.logicalFieldKeyFromDetailKeyForGroup(\r\n        d.fieldKey,\r\n        sectionIds,\r\n      );\r\n      return !customKeys.has(logicalKey);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Risolve la chiave logica del campo rispetto a tutte le sezioni del gruppo (campi flat o repeatable).\r\n   */\r\n  private logicalFieldKeyFromDetailKeyForGroup(\r\n    fieldKey: string,\r\n    sectionIds: string[],\r\n  ): string {\r\n    for (const id of sectionIds) {\r\n      const k = this.logicalFieldKeyFromDetailKey(fieldKey, id);\r\n      if (k !== fieldKey) return k;\r\n    }\r\n    return fieldKey;\r\n  }\r\n\r\n  /** Estrae la chiave campo da `fieldKey` flat o `sectionId[i].fieldKey`. */\r\n  private logicalFieldKeyFromDetailKey(\r\n    fieldKey: string,\r\n    sectionId: string,\r\n  ): string {\r\n    const re = new RegExp(`^${this.escapeRegExp(sectionId)}\\\\[\\\\d+\\\\]\\\\.(.+)$`);\r\n    const m = fieldKey.match(re);\r\n    return m ? m[1] : fieldKey;\r\n  }\r\n\r\n  private escapeRegExp(s: string): string {\r\n    return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\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.wizardStepGroups.map((g) => ({\r\n        title: g[0]?.wizardStep?.stepTitle || g[0]?.title,\r\n        description: g[0]?.wizardStep?.stepDescription || g[0]?.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"]}
@@ -3,4 +3,4 @@
3
3
  * Tipi per lo schema del form, sezioni e configurazione.
4
4
  */
5
5
  export {};
6
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema.types.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/form-builder/types/schema.types.ts"],"names":[],"mappings":"AAAA;;;GAGG","sourcesContent":["/**\n * @module ng-ui-system/form-builder\n * Tipi per lo schema del form, sezioni e configurazione.\n */\n\nimport { UiFormFieldDescriptor } from './field.types';\nimport { UiFieldCondition } from './condition.types';\nimport { UiIconName } from '../../../core/types';\n\n// ─── Wizard step ──────────────────────────────────────────────────\n\n/**\n * Configurazione di validazione per un singolo step del wizard.\n */\nexport interface UiWizardStepValidation {\n  /** Se lo step richiede validazione prima di procedere. @default true */\n  required?: boolean;\n  /** Campi specifici da validare in questo step (se omesso, tutti i campi dello step). */\n  validateFields?: string[];\n  /**\n   * Se `false`, nella navigazione wizard i campi `type: 'custom'` non concorrono\n   * alla validità dello step (e i loro errori non bloccano il passaggio).\n   * @default true\n   */\n  validateCustomComponents?: boolean;\n}\n\n/**\n * Funzione di validazione custom per uno step del wizard.\n * Viene invocata prima della navigazione allo step successivo.\n *\n * @param formData - Dati correnti del form completo.\n * @param currentStep - Indice dello step corrente (0-based).\n * @param isValid - Risultato della validazione standard del form.\n * @returns `true` per permettere la navigazione, `false` per bloccarla.\n */\nexport type UiWizardValidationFn = (\n  formData: Record<string, any>,\n  currentStep: number,\n  isValid: boolean,\n) => Promise<boolean> | boolean;\n\n/**\n * Configurazione wizard per una sezione del form.\n */\nexport interface UiWizardStepConfig {\n  /** Numero dello step (1-based per UI, convertito a 0-based internamente). */\n  stepNumber: number;\n  /** Titolo personalizzato per lo step. */\n  stepTitle?: string;\n  /** Descrizione dello step. */\n  stepDescription?: string;\n  /** Configurazione di validazione per questo step. */\n  validation?: UiWizardStepValidation;\n  /** Permette di saltare questo step senza validazione. */\n  allowSkip?: boolean;\n  /** Nome del metodo di validazione custom (deve essere registrato nel wizard). */\n  customValidationFn?: string;\n  /** Metadati aggiuntivi per lo step. */\n  metadata?: Record<string, any>;\n}\n\n// ─── Sezione ripetibile ───────────────────────────────────────────\n\n/**\n * Configurazione per sezioni ripetibili (dynamic repeater).\n * Permette di aggiungere/rimuovere/duplicare N istanze di un blocco\n * di campi, con il modello reattivo gestito tramite FormArray.\n *\n * @example\n * ```typescript\n * const section: UiFormSection = {\n *   id: 'obiettivi',\n *   title: 'Obiettivi formativi',\n *   repeatable: {\n *     min: 1,\n *     max: 10,\n *     addLabel: 'Aggiungi obiettivo',\n *     addIcon: 'plus',\n *     instanceTitle: 'Obiettivo {index}',\n *   },\n *   fields: [ ... ],\n * };\n * ```\n */\nexport interface UiRepeatableConfig {\n  /**\n   * Numero minimo di istanze obbligatorie.\n   * Se `min > 0`, alla creazione del form vengono pre-create almeno `min` istanze\n   * (vuote o da `initialData[sectionId]`). Se omesso, vale `1`.\n   * Con `min: 0` non si aggiungono righe oltre a quelle già presenti in `initialData`.\n   */\n  min?: number;\n  /** Numero massimo di istanze (undefined = illimitato). */\n  max?: number;\n  /** Label per il pulsante \"Aggiungi\". @default '+ Aggiungi' */\n  addLabel?: string;\n  /** Icona Lucide per il pulsante \"Aggiungi\". @default 'plus' */\n  addIcon?: UiIconName;\n  /** Template per il titolo di ogni istanza. `{index}` viene sostituito con l'indice 1-based. */\n  instanceTitle?: string;\n  /** Se richiedere conferma prima della rimozione. @default false */\n  confirmRemove?: boolean;\n  /** Messaggio di conferma rimozione. @default 'Sei sicuro di voler rimuovere questa sezione?' */\n  confirmMessage?: string;\n}\n\n// ─── Sezione ──────────────────────────────────────────────────────\n\n/**\n * Sezione del form che raggruppa campi correlati.\n * Supporta collapsibilita (accordion), condizioni di visibilita,\n * configurazione wizard opzionale e ripetibilita dinamica.\n *\n * @example\n * ```typescript\n * const section: UiFormSection = {\n *   id: 'anagrafica',\n *   title: 'Dati anagrafici',\n *   description: 'Informazioni personali del candidato',\n *   collapsible: true,\n *   fields: [ ... ],\n * };\n * ```\n */\nexport interface UiFormSection {\n  /** ID univoco della sezione. */\n  id: string;\n  /** Titolo della sezione. */\n  title?: string;\n  /** Descrizione della sezione. */\n  description?: string;\n  /** Icona Lucide opzionale per la sezione (visibile in modalità collapsible). @see https://lucide.dev/icons */\n  icon?: UiIconName;\n  /** Campi contenuti nella sezione. */\n  fields: UiFormFieldDescriptor[];\n  /** Condizioni per la visibilita della sezione. */\n  conditions?: UiFieldCondition[];\n  /** Se la sezione e collassabile (accordion). */\n  collapsible?: boolean;\n  /** Se la sezione e inizialmente collassata. @default false */\n  collapsed?: boolean;\n  /** Classi CSS aggiuntive per la sezione. */\n  cssClasses?: string[];\n  /** Configurazione wizard per questa sezione. */\n  wizardStep?: UiWizardStepConfig;\n  /** Configurazione per sezione ripetibile. Se definito, la sezione diventa un dynamic repeater. */\n  repeatable?: UiRepeatableConfig;\n}\n\n// ─── Configurazione form ──────────────────────────────────────────\n\n/**\n * Configurazione globale del form builder.\n */\nexport interface UiFormBuilderConfig {\n  /** Layout del form. @default 'grid' */\n  layout?: 'vertical' | 'horizontal' | 'grid';\n  /** Numero di colonne nel grid. @default 12 */\n  columns?: number;\n  /** Mostra i pulsanti di azione (submit/reset). @default true */\n  showButtons?: boolean;\n  /** Nasconde l'intero footer (pulsanti + error summary). */\n  hideFooter?: boolean;\n  /** Label personalizzate per i pulsanti. */\n  buttonLabels?: {\n    submit?: string;\n    reset?: string;\n    cancel?: string;\n  };\n  /** Classi CSS globali sul container del form. */\n  cssClasses?: string[];\n}\n\n// ─── Configurazione wizard ────────────────────────────────────────\n\n/**\n * Configurazione per la modalita wizard del form.\n * Usata dal componente `UiFormWizardComponent`.\n *\n * @example\n * ```typescript\n * const wizardConfig: UiWizardConfig = {\n *   showProgress: true,\n *   progressStyle: 'steps',\n *   allowBackNavigation: true,\n *   buttonLabels: { next: 'Avanti', previous: 'Indietro', finish: 'Completa' },\n * };\n * ```\n */\nexport interface UiWizardConfig {\n  /** ID per sincronizzazione wireless tra istanze. */\n  wizardId?: string;\n  /** Nasconde l'header interno del wizard per consentire un header esterno. */\n  wirelessHeading?: boolean;\n  /** Mostra la barra di progresso. @default true */\n  showProgress?: boolean;\n  /** Stile della barra di progresso. @default 'steps' */\n  progressStyle?: 'linear' | 'steps';\n  /** Permette navigazione all'indietro. @default true */\n  allowBackNavigation?: boolean;\n  /** Permette navigazione in avanti senza validazione. @default false */\n  allowForwardNavigation?: boolean;\n  /** Salva automaticamente i dati ad ogni cambio step. */\n  autoSave?: boolean;\n  /** Callback per il salvataggio dei dati dello step. */\n  onStepSave?: (stepData: Record<string, any>, stepNumber: number) => Promise<void> | void;\n  /** Mostra pulsanti di navigazione (precedente/successivo). @default true */\n  showNavigationButtons?: boolean;\n  /** Label personalizzate per i pulsanti del wizard. */\n  buttonLabels?: {\n    previous?: string;\n    next?: string;\n    finish?: string;\n    save?: string;\n  };\n}\n\n/**\n * Stato corrente del wizard, esposto dal `WizardSyncService`.\n */\nexport interface UiWizardState {\n  /** Step corrente (0-based). */\n  currentStep: number;\n  /** Numero totale di step. */\n  totalSteps: number;\n  /** Indici degli step completati. */\n  completedSteps: Set<number>;\n  /** Indici degli step visitati. */\n  visitedSteps: Set<number>;\n  /** Se il form nello step corrente e valido. */\n  isValid: boolean;\n  /** Errori per step. */\n  stepErrors: Record<number, string[]>;\n}\n\n// ─── Schema ───────────────────────────────────────────────────────\n\n/**\n * Schema completo del form.\n *\n * Rappresenta la configurazione dichiarativa dell'intero form:\n * sezioni, campi, validazioni, condizioni e impostazioni globali.\n *\n * @example\n * ```typescript\n * export const myForm: UiFormSchema = {\n *   id: 'user-registration',\n *   title: 'Registrazione utente',\n *   sections: [\n *     { id: 'personal', title: 'Dati personali', fields: [...] },\n *     { id: 'address', title: 'Indirizzo', fields: [...], collapsible: true },\n *   ],\n *   config: { columns: 12, showButtons: true },\n * };\n * ```\n */\nexport interface UiFormSchema {\n  /** ID univoco del form. */\n  id: string;\n  /** Titolo del form. */\n  title?: string;\n  /** Descrizione del form. */\n  description?: string;\n  /** Sezioni del form. */\n  sections: UiFormSection[];\n  /** Configurazione globale. */\n  config?: UiFormBuilderConfig;\n}\n\n// ─── Tipi di utilita ──────────────────────────────────────────────\n\n/**\n * Dati del form come mappa chiave-valore.\n */\nexport type UiFormData = Record<string, any>;\n\n/**\n * Evento emesso dai campi custom.\n */\nexport interface UiFormCustomEvent {\n  /** Chiave del campo che ha emesso l'evento. */\n  field: string;\n  /** Nome dell'evento. */\n  event: string;\n  /** Dati associati all'evento. */\n  data: any;\n}\n\n/**\n * Evento emesso al cambio step del wizard.\n */\nexport interface UiWizardStepChangeEvent {\n  /** Indice dello step di partenza. */\n  from: number;\n  /** Indice dello step di destinazione. */\n  to: number;\n  /** Direzione della navigazione. */\n  direction: 'next' | 'previous';\n}\n\n/**\n * Evento emesso al completamento di uno step.\n */\nexport interface UiWizardStepCompleteEvent {\n  /** Indice dello step completato. */\n  step: number;\n  /** Dati del form al momento del completamento. */\n  data: Record<string, any>;\n}\n"]}
6
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema.types.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/form-builder/types/schema.types.ts"],"names":[],"mappings":"AAAA;;;GAGG","sourcesContent":["/**\n * @module ng-ui-system/form-builder\n * Tipi per lo schema del form, sezioni e configurazione.\n */\n\nimport { UiFormFieldDescriptor } from './field.types';\nimport { UiFieldCondition } from './condition.types';\nimport { UiIconName } from '../../../core/types';\n\n// ─── Wizard step ──────────────────────────────────────────────────\n\n/**\n * Configurazione di validazione per un singolo step del wizard.\n */\nexport interface UiWizardStepValidation {\n  /** Se lo step richiede validazione prima di procedere. @default true */\n  required?: boolean;\n  /** Campi specifici da validare in questo step (se omesso, tutti i campi dello step). */\n  validateFields?: string[];\n  /**\n   * Se `false`, nella navigazione wizard i campi `type: 'custom'` non concorrono\n   * alla validità dello step (e i loro errori non bloccano il passaggio).\n   * @default true\n   */\n  validateCustomComponents?: boolean;\n}\n\n/**\n * Funzione di validazione custom per uno step del wizard.\n * Viene invocata prima della navigazione allo step successivo.\n *\n * @param formData - Dati correnti del form completo.\n * @param currentStep - Indice dello step corrente (0-based).\n * @param isValid - Risultato della validazione standard del form.\n * @returns `true` per permettere la navigazione, `false` per bloccarla.\n */\nexport type UiWizardValidationFn = (\n  formData: Record<string, any>,\n  currentStep: number,\n  isValid: boolean,\n) => Promise<boolean> | boolean;\n\n/**\n * Configurazione wizard per una sezione del form.\n */\nexport interface UiWizardStepConfig {\n  /** Numero dello step (1-based per UI, convertito a 0-based internamente). */\n  stepNumber: number;\n  /** Titolo personalizzato per lo step. */\n  stepTitle?: string;\n  /** Descrizione dello step. */\n  stepDescription?: string;\n  /** Configurazione di validazione per questo step. */\n  validation?: UiWizardStepValidation;\n  /** Permette di saltare questo step senza validazione. */\n  allowSkip?: boolean;\n  /** Nome del metodo di validazione custom (deve essere registrato nel wizard). */\n  customValidationFn?: string;\n  /** Metadati aggiuntivi per lo step. */\n  metadata?: Record<string, any>;\n}\n\n// ─── Sezione ripetibile ───────────────────────────────────────────\n\n/**\n * Configurazione per sezioni ripetibili (dynamic repeater).\n * Permette di aggiungere/rimuovere/duplicare N istanze di un blocco\n * di campi, con il modello reattivo gestito tramite FormArray.\n *\n * @example\n * ```typescript\n * const section: UiFormSection = {\n *   id: 'obiettivi',\n *   title: 'Obiettivi formativi',\n *   repeatable: {\n *     min: 1,\n *     max: 10,\n *     addLabel: 'Aggiungi obiettivo',\n *     addIcon: 'plus',\n *     instanceTitle: 'Obiettivo {index}',\n *   },\n *   fields: [ ... ],\n * };\n * ```\n */\nexport interface UiRepeatableConfig {\n  /**\n   * Numero minimo di istanze obbligatorie.\n   * Se `min > 0`, alla creazione del form vengono pre-create almeno `min` istanze\n   * (vuote o da `initialData[sectionId]`). Se omesso, vale `1`.\n   * Con `min: 0` non si aggiungono righe oltre a quelle già presenti in `initialData`.\n   */\n  min?: number;\n  /** Numero massimo di istanze (undefined = illimitato). */\n  max?: number;\n  /** Label per il pulsante \"Aggiungi\". @default '+ Aggiungi' */\n  addLabel?: string;\n  /** Icona Lucide per il pulsante \"Aggiungi\". @default 'plus' */\n  addIcon?: UiIconName;\n  /** Template per il titolo di ogni istanza. `{index}` viene sostituito con l'indice 1-based. */\n  instanceTitle?: string;\n  /** Se richiedere conferma prima della rimozione. @default false */\n  confirmRemove?: boolean;\n  /** Messaggio di conferma rimozione. @default 'Sei sicuro di voler rimuovere questa sezione?' */\n  confirmMessage?: string;\n}\n\n// ─── Sezione ──────────────────────────────────────────────────────\n\n/**\n * Sezione del form che raggruppa campi correlati.\n * Supporta collapsibilita (accordion), condizioni di visibilita,\n * configurazione wizard opzionale e ripetibilita dinamica.\n *\n * @example\n * ```typescript\n * const section: UiFormSection = {\n *   id: 'anagrafica',\n *   title: 'Dati anagrafici',\n *   description: 'Informazioni personali del candidato',\n *   collapsible: true,\n *   fields: [ ... ],\n * };\n * ```\n */\nexport interface UiFormSection {\n  /** ID univoco della sezione. */\n  id: string;\n  /** Titolo della sezione. */\n  title?: string;\n  /** Descrizione della sezione. */\n  description?: string;\n  /** Icona Lucide opzionale per la sezione (visibile in modalità collapsible). @see https://lucide.dev/icons */\n  icon?: UiIconName;\n  /** Campi contenuti nella sezione. */\n  fields: UiFormFieldDescriptor[];\n  /** Condizioni per la visibilita della sezione. */\n  conditions?: UiFieldCondition[];\n  /** Se la sezione e collassabile (accordion). */\n  collapsible?: boolean;\n  /** Se la sezione e inizialmente collassata. @default false */\n  collapsed?: boolean;\n  /** Classi CSS aggiuntive per la sezione. */\n  cssClasses?: string[];\n  /**\n   * Configurazione wizard per questa sezione.\n   * Più sezioni possono condividere lo stesso `wizardStep.stepNumber`: formano un unico\n   * passo del wizard (es. più blocchi `collapsible` nello stesso step). Titolo, descrizione\n   * e validazione del passo si intendono dalla **prima** sezione del gruppo in ordine di schema.\n   * Le chiavi dei campi devono restare univoche tra tutte le sezioni dello stesso schema wizard.\n   */\n  wizardStep?: UiWizardStepConfig;\n  /** Configurazione per sezione ripetibile. Se definito, la sezione diventa un dynamic repeater. */\n  repeatable?: UiRepeatableConfig;\n}\n\n// ─── Configurazione form ──────────────────────────────────────────\n\n/**\n * Configurazione globale del form builder.\n */\nexport interface UiFormBuilderConfig {\n  /** Layout del form. @default 'grid' */\n  layout?: 'vertical' | 'horizontal' | 'grid';\n  /** Numero di colonne nel grid. @default 12 */\n  columns?: number;\n  /** Mostra i pulsanti di azione (submit/reset). @default true */\n  showButtons?: boolean;\n  /** Nasconde l'intero footer (pulsanti + error summary). */\n  hideFooter?: boolean;\n  /** Label personalizzate per i pulsanti. */\n  buttonLabels?: {\n    submit?: string;\n    reset?: string;\n    cancel?: string;\n  };\n  /** Classi CSS globali sul container del form. */\n  cssClasses?: string[];\n}\n\n// ─── Configurazione wizard ────────────────────────────────────────\n\n/**\n * Configurazione per la modalita wizard del form.\n * Usata dal componente `UiFormWizardComponent`.\n *\n * @example\n * ```typescript\n * const wizardConfig: UiWizardConfig = {\n *   showProgress: true,\n *   progressStyle: 'steps',\n *   allowBackNavigation: true,\n *   buttonLabels: { next: 'Avanti', previous: 'Indietro', finish: 'Completa' },\n * };\n * ```\n */\nexport interface UiWizardConfig {\n  /** ID per sincronizzazione wireless tra istanze. */\n  wizardId?: string;\n  /** Nasconde l'header interno del wizard per consentire un header esterno. */\n  wirelessHeading?: boolean;\n  /** Mostra la barra di progresso. @default true */\n  showProgress?: boolean;\n  /** Stile della barra di progresso. @default 'steps' */\n  progressStyle?: 'linear' | 'steps';\n  /** Permette navigazione all'indietro. @default true */\n  allowBackNavigation?: boolean;\n  /** Permette navigazione in avanti senza validazione. @default false */\n  allowForwardNavigation?: boolean;\n  /** Salva automaticamente i dati ad ogni cambio step. */\n  autoSave?: boolean;\n  /** Callback per il salvataggio dei dati dello step. */\n  onStepSave?: (stepData: Record<string, any>, stepNumber: number) => Promise<void> | void;\n  /** Mostra pulsanti di navigazione (precedente/successivo). @default true */\n  showNavigationButtons?: boolean;\n  /** Label personalizzate per i pulsanti del wizard. */\n  buttonLabels?: {\n    previous?: string;\n    next?: string;\n    finish?: string;\n    save?: string;\n  };\n}\n\n/**\n * Stato corrente del wizard, esposto dal `WizardSyncService`.\n */\nexport interface UiWizardState {\n  /** Step corrente (0-based). */\n  currentStep: number;\n  /** Numero totale di step. */\n  totalSteps: number;\n  /** Indici degli step completati. */\n  completedSteps: Set<number>;\n  /** Indici degli step visitati. */\n  visitedSteps: Set<number>;\n  /** Se il form nello step corrente e valido. */\n  isValid: boolean;\n  /** Errori per step. */\n  stepErrors: Record<number, string[]>;\n}\n\n// ─── Schema ───────────────────────────────────────────────────────\n\n/**\n * Schema completo del form.\n *\n * Rappresenta la configurazione dichiarativa dell'intero form:\n * sezioni, campi, validazioni, condizioni e impostazioni globali.\n *\n * @example\n * ```typescript\n * export const myForm: UiFormSchema = {\n *   id: 'user-registration',\n *   title: 'Registrazione utente',\n *   sections: [\n *     { id: 'personal', title: 'Dati personali', fields: [...] },\n *     { id: 'address', title: 'Indirizzo', fields: [...], collapsible: true },\n *   ],\n *   config: { columns: 12, showButtons: true },\n * };\n * ```\n */\nexport interface UiFormSchema {\n  /** ID univoco del form. */\n  id: string;\n  /** Titolo del form. */\n  title?: string;\n  /** Descrizione del form. */\n  description?: string;\n  /** Sezioni del form. */\n  sections: UiFormSection[];\n  /** Configurazione globale. */\n  config?: UiFormBuilderConfig;\n}\n\n// ─── Tipi di utilita ──────────────────────────────────────────────\n\n/**\n * Dati del form come mappa chiave-valore.\n */\nexport type UiFormData = Record<string, any>;\n\n/**\n * Evento emesso dai campi custom.\n */\nexport interface UiFormCustomEvent {\n  /** Chiave del campo che ha emesso l'evento. */\n  field: string;\n  /** Nome dell'evento. */\n  event: string;\n  /** Dati associati all'evento. */\n  data: any;\n}\n\n/**\n * Evento emesso al cambio step del wizard.\n */\nexport interface UiWizardStepChangeEvent {\n  /** Indice dello step di partenza. */\n  from: number;\n  /** Indice dello step di destinazione. */\n  to: number;\n  /** Direzione della navigazione. */\n  direction: 'next' | 'previous';\n}\n\n/**\n * Evento emesso al completamento di uno step.\n */\nexport interface UiWizardStepCompleteEvent {\n  /** Indice dello step completato. */\n  step: number;\n  /** Dati del form al momento del completamento. */\n  data: Record<string, any>;\n}\n"]}