@gnggln/ng-ui-system 1.0.0-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/gnggln-ng-ui-system.mjs +5 -0
- package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
- package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
- package/esm2022/lib/components/accordion/index.mjs +2 -0
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
- package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
- package/esm2022/lib/components/base-layout/index.mjs +14 -0
- package/esm2022/lib/components/button/button-area.component.mjs +196 -0
- package/esm2022/lib/components/button/button.component.mjs +164 -0
- package/esm2022/lib/components/button/button.types.mjs +6 -0
- package/esm2022/lib/components/button/index.mjs +16 -0
- package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
- package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
- package/esm2022/lib/components/crud-table/index.mjs +16 -0
- package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
- package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
- package/esm2022/lib/components/form-builder/index.mjs +19 -0
- package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
- package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
- package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
- package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
- package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
- package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
- package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
- package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
- package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
- package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
- package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
- package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
- package/esm2022/lib/components/layout-builder/index.mjs +18 -0
- package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
- package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
- package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
- package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
- package/esm2022/lib/components/modal/index.mjs +4 -0
- package/esm2022/lib/components/modal/modal.component.mjs +139 -0
- package/esm2022/lib/components/modal/modal.service.mjs +194 -0
- package/esm2022/lib/components/modal/modal.types.mjs +6 -0
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
- package/esm2022/lib/components/page-header/index.mjs +20 -0
- package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
- package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
- package/esm2022/lib/components/table/index.mjs +2 -0
- package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
- package/esm2022/lib/components/table/table.types.mjs +6 -0
- package/esm2022/lib/core/types/index.mjs +6 -0
- package/esm2022/lib/core/utils/index.mjs +53 -0
- package/esm2022/lib/sources/location-data.opt.json +8942 -0
- package/esm2022/lib/sources/nazioni.opt.json +215 -0
- package/esm2022/public-api.mjs +34 -0
- package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/accordion/accordion.component.d.ts +118 -0
- package/lib/components/accordion/accordion.types.d.ts +62 -0
- package/lib/components/accordion/index.d.ts +2 -0
- package/lib/components/base-layout/base-layout.component.d.ts +83 -0
- package/lib/components/base-layout/base-layout.types.d.ts +26 -0
- package/lib/components/base-layout/index.d.ts +13 -0
- package/lib/components/button/button-area.component.d.ts +88 -0
- package/lib/components/button/button.component.d.ts +55 -0
- package/lib/components/button/button.types.d.ts +70 -0
- package/lib/components/button/index.d.ts +15 -0
- package/lib/components/crud-table/crud-table.component.d.ts +143 -0
- package/lib/components/crud-table/crud-table.types.d.ts +207 -0
- package/lib/components/crud-table/index.d.ts +15 -0
- package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
- package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
- package/lib/components/form-builder/form-builder.component.d.ts +183 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
- package/lib/components/form-builder/index.d.ts +13 -0
- package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
- package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
- package/lib/components/form-builder/services/location.service.d.ts +83 -0
- package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
- package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
- package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
- package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
- package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
- package/lib/components/form-builder/types/condition.types.d.ts +51 -0
- package/lib/components/form-builder/types/field.types.d.ts +288 -0
- package/lib/components/form-builder/types/index.d.ts +5 -0
- package/lib/components/form-builder/types/schema.types.d.ts +227 -0
- package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
- package/lib/components/form-builder/types/validation.types.d.ts +174 -0
- package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
- package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
- package/lib/components/form-builder-editor/index.d.ts +15 -0
- package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
- package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
- package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
- package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
- package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
- package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
- package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
- package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
- package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
- package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
- package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
- package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
- package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
- package/lib/components/layout-builder/index.d.ts +16 -0
- package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
- package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
- package/lib/components/layout-builder/layout.service.d.ts +100 -0
- package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
- package/lib/components/modal/index.d.ts +4 -0
- package/lib/components/modal/modal.component.d.ts +44 -0
- package/lib/components/modal/modal.service.d.ts +93 -0
- package/lib/components/modal/modal.types.d.ts +110 -0
- package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
- package/lib/components/page-header/index.d.ts +16 -0
- package/lib/components/page-header/page-header.component.d.ts +59 -0
- package/lib/components/page-header/page-header.types.d.ts +96 -0
- package/lib/components/table/index.d.ts +2 -0
- package/lib/components/table/paginated-table.component.d.ts +85 -0
- package/lib/components/table/table.types.d.ts +81 -0
- package/lib/core/types/index.d.ts +57 -0
- package/lib/core/utils/index.d.ts +29 -0
- package/package.json +44 -0
- package/public-api.d.ts +22 -0
|
@@ -0,0 +1,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
|