@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,730 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/form-builder-editor
|
|
3
|
+
* Componente principale dell'editor visuale del form builder.
|
|
4
|
+
*
|
|
5
|
+
* Layout a tre pannelli:
|
|
6
|
+
* - Sinistra: palette dei tipi di campo (drag source)
|
|
7
|
+
* - Centro: sezioni con campi (drop target + editor inline)
|
|
8
|
+
* - Destra: pannello di configurazione del campo selezionato
|
|
9
|
+
*
|
|
10
|
+
* Funzionalita principali:
|
|
11
|
+
* - Drag & drop dei campi dalla palette alle sezioni
|
|
12
|
+
* - Riordino campi all'interno e tra sezioni
|
|
13
|
+
* - Configurazione campi in tempo reale
|
|
14
|
+
* - Preview live del form
|
|
15
|
+
* - Import/export JSON
|
|
16
|
+
* - Auto-save su localStorage
|
|
17
|
+
*/
|
|
18
|
+
import { Component, ChangeDetectionStrategy, ChangeDetectorRef, inject, Optional, Inject, } from '@angular/core';
|
|
19
|
+
import { DragDropModule } from '@angular/cdk/drag-drop';
|
|
20
|
+
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
21
|
+
import { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';
|
|
22
|
+
import { MatTabsModule } from '@angular/material/tabs';
|
|
23
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
24
|
+
import { Subject, interval } from 'rxjs';
|
|
25
|
+
import { takeUntil } from 'rxjs/operators';
|
|
26
|
+
import { UiEditorStateService } from './services/editor-state.service';
|
|
27
|
+
import { UiEditorPersistenceService } from './services/editor-persistence.service';
|
|
28
|
+
import { UiEditorFieldFactoryService } from './services/field-factory.service';
|
|
29
|
+
// Sotto-componenti
|
|
30
|
+
import { UiEditorToolbarComponent } from './sub-components/editor-toolbar/editor-toolbar.component';
|
|
31
|
+
import { UiFieldPaletteComponent } from './sub-components/field-palette/field-palette.component';
|
|
32
|
+
import { UiSectionEditorComponent } from './sub-components/section-editor/section-editor.component';
|
|
33
|
+
import { UiEditorFieldConfigPanelComponent } from './sub-components/field-config-panel/field-config-panel.component';
|
|
34
|
+
import { UiEditorPreviewContainerComponent } from './sub-components/preview-container/preview-container.component';
|
|
35
|
+
import { UiButtonComponent } from '../button/index';
|
|
36
|
+
import * as i0 from "@angular/core";
|
|
37
|
+
import * as i1 from "@angular/material/dialog";
|
|
38
|
+
import * as i2 from "@angular/cdk/drag-drop";
|
|
39
|
+
import * as i3 from "@angular/material/tabs";
|
|
40
|
+
import * as i4 from "lucide-angular";
|
|
41
|
+
export class UiFormBuilderEditorComponent {
|
|
42
|
+
constructor(dialogRef, dialogData) {
|
|
43
|
+
this.dialogRef = dialogRef;
|
|
44
|
+
this.dialogData = dialogData;
|
|
45
|
+
/** @internal Servizio per lo stato dell'editor. */
|
|
46
|
+
this.stateService = inject(UiEditorStateService);
|
|
47
|
+
/** @internal Servizio per la persistenza. */
|
|
48
|
+
this.persistenceService = inject(UiEditorPersistenceService);
|
|
49
|
+
/** @internal Factory per i campi. */
|
|
50
|
+
this.fieldFactory = inject(UiEditorFieldFactoryService);
|
|
51
|
+
/** @internal Snack bar per i messaggi. */
|
|
52
|
+
this.snackBar = inject(MatSnackBar);
|
|
53
|
+
/** @internal Change detector. */
|
|
54
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
55
|
+
/** @internal Subject per la distruzione. */
|
|
56
|
+
this.destroy$ = new Subject();
|
|
57
|
+
/** Indice del tab selezionato (0=Editor, 1=Preview). */
|
|
58
|
+
this.selectedTabIndex = 0;
|
|
59
|
+
/** Breadcrumb di navigazione. */
|
|
60
|
+
this.breadcrumb = [];
|
|
61
|
+
/** Se il componente e aperto in modalita dialog. */
|
|
62
|
+
this.isDialogMode = false;
|
|
63
|
+
/** Chiave per forzare il re-render del preview. */
|
|
64
|
+
this.previewKey = 0;
|
|
65
|
+
/** Campi disponibili per le condizioni (cache). */
|
|
66
|
+
this.availableFields = [];
|
|
67
|
+
/** Larghezza della sidebar sinistra (px). */
|
|
68
|
+
this.leftSidebarWidth = 280;
|
|
69
|
+
/** Larghezza della sidebar destra (px). */
|
|
70
|
+
this.rightSidebarWidth = 460;
|
|
71
|
+
/** Se e in corso il resize della sidebar destra. */
|
|
72
|
+
this.isResizing = false;
|
|
73
|
+
/** Se e in corso il resize della sidebar sinistra. */
|
|
74
|
+
this.isResizingLeft = false;
|
|
75
|
+
/** @internal Limiti di resize sidebar destra. */
|
|
76
|
+
this.MIN_SIDEBAR_WIDTH = 400;
|
|
77
|
+
this.MAX_SIDEBAR_WIDTH = 1200;
|
|
78
|
+
/** @internal Limiti di resize sidebar sinistra. */
|
|
79
|
+
this.MIN_LEFT_SIDEBAR_WIDTH = 200;
|
|
80
|
+
this.MAX_LEFT_SIDEBAR_WIDTH = 500;
|
|
81
|
+
this.resizeStartX = 0;
|
|
82
|
+
this.resizeStartWidth = 0;
|
|
83
|
+
this.SIDEBAR_WIDTH_KEY = 'ui-editor-right-sidebar-width';
|
|
84
|
+
this.LEFT_SIDEBAR_WIDTH_KEY = 'ui-editor-left-sidebar-width';
|
|
85
|
+
this.onResizeMove = (event) => {
|
|
86
|
+
if (!this.isResizing)
|
|
87
|
+
return;
|
|
88
|
+
const delta = this.resizeStartX - event.clientX;
|
|
89
|
+
this.rightSidebarWidth = Math.max(this.MIN_SIDEBAR_WIDTH, Math.min(this.MAX_SIDEBAR_WIDTH, this.resizeStartWidth + delta));
|
|
90
|
+
this.cdr.markForCheck();
|
|
91
|
+
};
|
|
92
|
+
this.onResizeEnd = () => {
|
|
93
|
+
if (!this.isResizing)
|
|
94
|
+
return;
|
|
95
|
+
this.isResizing = false;
|
|
96
|
+
document.removeEventListener('mousemove', this.onResizeMove);
|
|
97
|
+
document.removeEventListener('mouseup', this.onResizeEnd);
|
|
98
|
+
document.body.style.cursor = '';
|
|
99
|
+
document.body.style.userSelect = '';
|
|
100
|
+
this.saveSidebarWidth();
|
|
101
|
+
};
|
|
102
|
+
this.onLeftResizeMove = (event) => {
|
|
103
|
+
if (!this.isResizingLeft)
|
|
104
|
+
return;
|
|
105
|
+
const delta = event.clientX - this.resizeStartX;
|
|
106
|
+
this.leftSidebarWidth = Math.max(this.MIN_LEFT_SIDEBAR_WIDTH, Math.min(this.MAX_LEFT_SIDEBAR_WIDTH, this.resizeStartWidth + delta));
|
|
107
|
+
this.cdr.markForCheck();
|
|
108
|
+
};
|
|
109
|
+
this.onLeftResizeEnd = () => {
|
|
110
|
+
if (!this.isResizingLeft)
|
|
111
|
+
return;
|
|
112
|
+
this.isResizingLeft = false;
|
|
113
|
+
document.removeEventListener('mousemove', this.onLeftResizeMove);
|
|
114
|
+
document.removeEventListener('mouseup', this.onLeftResizeEnd);
|
|
115
|
+
document.body.style.cursor = '';
|
|
116
|
+
document.body.style.userSelect = '';
|
|
117
|
+
this.saveLeftSidebarWidth();
|
|
118
|
+
};
|
|
119
|
+
this.state = this.stateService.getCurrentState();
|
|
120
|
+
this.isDialogMode = !!this.dialogRef;
|
|
121
|
+
// Carica schema iniziale se fornito via dialog data
|
|
122
|
+
if (this.dialogData?.initialSchema) {
|
|
123
|
+
this.stateService.setSchema(this.dialogData.initialSchema);
|
|
124
|
+
}
|
|
125
|
+
this.loadSidebarWidth();
|
|
126
|
+
}
|
|
127
|
+
ngOnInit() {
|
|
128
|
+
// Sottoscrizione allo stato
|
|
129
|
+
this.stateService.state$.pipe(takeUntil(this.destroy$)).subscribe((state) => {
|
|
130
|
+
this.state = state;
|
|
131
|
+
this.updateBreadcrumb();
|
|
132
|
+
this.updateAvailableFields();
|
|
133
|
+
this.cdr.markForCheck();
|
|
134
|
+
});
|
|
135
|
+
// Controlla salvataggi precedenti da recuperare
|
|
136
|
+
this.checkForRecovery();
|
|
137
|
+
// Auto-save ogni 30 secondi
|
|
138
|
+
interval(30000)
|
|
139
|
+
.pipe(takeUntil(this.destroy$))
|
|
140
|
+
.subscribe(() => {
|
|
141
|
+
if (this.state.isDirty) {
|
|
142
|
+
this.autoSave();
|
|
143
|
+
}
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
ngOnDestroy() {
|
|
147
|
+
this.destroy$.next();
|
|
148
|
+
this.destroy$.complete();
|
|
149
|
+
}
|
|
150
|
+
// ─── Toolbar actions ─────────────────────────────────────────
|
|
151
|
+
/** Aggiunge una nuova sezione. */
|
|
152
|
+
onAddSection() {
|
|
153
|
+
this.stateService.addSection();
|
|
154
|
+
this.showMessage('Sezione aggiunta');
|
|
155
|
+
}
|
|
156
|
+
/** Salva nel localStorage. */
|
|
157
|
+
onSave() {
|
|
158
|
+
try {
|
|
159
|
+
this.persistenceService.saveToLocalStorage(this.state.schema);
|
|
160
|
+
this.stateService.markAsSaved();
|
|
161
|
+
this.showMessage('Salvato con successo');
|
|
162
|
+
}
|
|
163
|
+
catch (error) {
|
|
164
|
+
this.showMessage(error.message || 'Errore durante il salvataggio', true);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
/** Crea un nuovo form. */
|
|
168
|
+
onNewForm() {
|
|
169
|
+
if (this.state.isDirty && !confirm('Ci sono modifiche non salvate. Continuare?')) {
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
this.stateService.reset();
|
|
173
|
+
this.showMessage('Nuovo form creato');
|
|
174
|
+
}
|
|
175
|
+
/** Importa da file JSON. */
|
|
176
|
+
onImportJson() {
|
|
177
|
+
const input = document.createElement('input');
|
|
178
|
+
input.type = 'file';
|
|
179
|
+
input.accept = '.json';
|
|
180
|
+
input.onchange = async (e) => {
|
|
181
|
+
const file = e.target.files[0];
|
|
182
|
+
if (!file)
|
|
183
|
+
return;
|
|
184
|
+
try {
|
|
185
|
+
const schema = await this.persistenceService.importFromJSON(file);
|
|
186
|
+
if (this.state.isDirty && !confirm('Ci sono modifiche non salvate. Sovrascrivere?')) {
|
|
187
|
+
return;
|
|
188
|
+
}
|
|
189
|
+
this.stateService.setSchema(schema);
|
|
190
|
+
this.showMessage('Schema importato con successo');
|
|
191
|
+
}
|
|
192
|
+
catch (error) {
|
|
193
|
+
this.showMessage(error.message || "Errore durante l'import", true);
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
input.click();
|
|
197
|
+
}
|
|
198
|
+
/** Esporta come file JSON. */
|
|
199
|
+
onExportJson() {
|
|
200
|
+
try {
|
|
201
|
+
const validation = this.stateService.validateSchema();
|
|
202
|
+
if (!validation.valid) {
|
|
203
|
+
const message = `Schema con errori:\n${validation.errors.map((e) => e.message).join('\n')}`;
|
|
204
|
+
if (!confirm(`${message}\n\nEsportare comunque?`))
|
|
205
|
+
return;
|
|
206
|
+
}
|
|
207
|
+
this.persistenceService.exportToJSON(this.state.schema);
|
|
208
|
+
this.showMessage('Schema esportato con successo');
|
|
209
|
+
}
|
|
210
|
+
catch (error) {
|
|
211
|
+
this.showMessage(error.message || "Errore durante l'export", true);
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
/** Pulisce tutto lo schema. */
|
|
215
|
+
onClear() {
|
|
216
|
+
if (!confirm('Vuoi davvero eliminare tutto? Questa azione non puo essere annullata.'))
|
|
217
|
+
return;
|
|
218
|
+
this.persistenceService.createBackup(this.state.schema);
|
|
219
|
+
this.stateService.clearSchema();
|
|
220
|
+
this.showMessage('Schema pulito');
|
|
221
|
+
}
|
|
222
|
+
// ─── Sezioni ─────────────────────────────────────────────────
|
|
223
|
+
onSectionDuplicate(sectionId) {
|
|
224
|
+
this.stateService.duplicateSection(sectionId);
|
|
225
|
+
this.showMessage('Sezione duplicata');
|
|
226
|
+
}
|
|
227
|
+
onSectionDelete(sectionId) {
|
|
228
|
+
if (!confirm('Vuoi davvero eliminare questa sezione?'))
|
|
229
|
+
return;
|
|
230
|
+
this.stateService.removeSection(sectionId);
|
|
231
|
+
this.showMessage('Sezione eliminata');
|
|
232
|
+
}
|
|
233
|
+
// ─── Campi ───────────────────────────────────────────────────
|
|
234
|
+
onFieldSelect(sectionId, fieldKey) {
|
|
235
|
+
this.stateService.selectField(sectionId, fieldKey);
|
|
236
|
+
}
|
|
237
|
+
onFieldDuplicate(sectionId, fieldKey) {
|
|
238
|
+
this.stateService.duplicateField(sectionId, fieldKey);
|
|
239
|
+
this.showMessage('Campo duplicato');
|
|
240
|
+
}
|
|
241
|
+
onFieldDelete(sectionId, fieldKey) {
|
|
242
|
+
if (!confirm('Vuoi davvero eliminare questo campo?'))
|
|
243
|
+
return;
|
|
244
|
+
this.stateService.removeField(sectionId, fieldKey);
|
|
245
|
+
this.showMessage('Campo eliminato');
|
|
246
|
+
}
|
|
247
|
+
onFieldUpdate(sectionId, fieldKey, updates) {
|
|
248
|
+
this.stateService.updateField(sectionId, fieldKey, updates);
|
|
249
|
+
}
|
|
250
|
+
/** Gestisce il drop di un campo (dalla palette o tra sezioni). */
|
|
251
|
+
onFieldDrop(event) {
|
|
252
|
+
const prevId = event.previousContainer.id;
|
|
253
|
+
if (prevId.startsWith('palette-')) {
|
|
254
|
+
// Drop dalla palette
|
|
255
|
+
const paletteItem = event.item.data;
|
|
256
|
+
const section = event.container.data;
|
|
257
|
+
const newField = this.fieldFactory.createDefaultField(paletteItem.type);
|
|
258
|
+
this.stateService.addFieldAtIndex(section.id, newField, event.currentIndex);
|
|
259
|
+
this.showMessage(`Campo ${paletteItem.label} aggiunto`);
|
|
260
|
+
}
|
|
261
|
+
else if (event.previousContainer !== event.container) {
|
|
262
|
+
// Spostamento tra sezioni
|
|
263
|
+
const field = event.item.data;
|
|
264
|
+
const fromSection = event.previousContainer.data;
|
|
265
|
+
const toSection = event.container.data;
|
|
266
|
+
this.stateService.moveFieldBetweenSections(fromSection.id, toSection.id, field.key, event.currentIndex);
|
|
267
|
+
this.showMessage('Campo spostato');
|
|
268
|
+
}
|
|
269
|
+
else {
|
|
270
|
+
// Riordino nella stessa sezione
|
|
271
|
+
const section = event.container.data;
|
|
272
|
+
this.stateService.moveFieldInSection(section.id, event.previousIndex, event.currentIndex);
|
|
273
|
+
}
|
|
274
|
+
this.cdr.markForCheck();
|
|
275
|
+
}
|
|
276
|
+
// ─── Computed properties ─────────────────────────────────────
|
|
277
|
+
/** Campo attualmente selezionato. */
|
|
278
|
+
get selectedField() {
|
|
279
|
+
if (!this.state.selectedFieldKey)
|
|
280
|
+
return null;
|
|
281
|
+
return this.stateService.findFieldByKey(this.state.selectedFieldKey);
|
|
282
|
+
}
|
|
283
|
+
/** ID della sezione del campo selezionato. */
|
|
284
|
+
get selectedFieldSectionId() {
|
|
285
|
+
if (!this.state.selectedFieldKey)
|
|
286
|
+
return null;
|
|
287
|
+
return this.stateService.findSectionByFieldKey(this.state.selectedFieldKey)?.id || null;
|
|
288
|
+
}
|
|
289
|
+
// ─── Tab / Preview ───────────────────────────────────────────
|
|
290
|
+
onTabChange(index) {
|
|
291
|
+
this.selectedTabIndex = index;
|
|
292
|
+
if (index === 1)
|
|
293
|
+
this.refreshPreview();
|
|
294
|
+
}
|
|
295
|
+
refreshPreview() {
|
|
296
|
+
this.previewKey++;
|
|
297
|
+
this.cdr.markForCheck();
|
|
298
|
+
}
|
|
299
|
+
// ─── Dialog ──────────────────────────────────────────────────
|
|
300
|
+
/** Chiude la modale restituendo lo schema corrente. */
|
|
301
|
+
closeDialog() {
|
|
302
|
+
this.dialogRef?.close(this.state.schema);
|
|
303
|
+
}
|
|
304
|
+
// ─── Resize sidebar ──────────────────────────────────────────
|
|
305
|
+
onResizeStart(event) {
|
|
306
|
+
event.preventDefault();
|
|
307
|
+
this.isResizing = true;
|
|
308
|
+
this.resizeStartX = event.clientX;
|
|
309
|
+
this.resizeStartWidth = this.rightSidebarWidth;
|
|
310
|
+
document.addEventListener('mousemove', this.onResizeMove);
|
|
311
|
+
document.addEventListener('mouseup', this.onResizeEnd);
|
|
312
|
+
document.body.style.cursor = 'ew-resize';
|
|
313
|
+
document.body.style.userSelect = 'none';
|
|
314
|
+
}
|
|
315
|
+
// ─── Resize sidebar sinistra ─────────────────────────────────
|
|
316
|
+
onLeftResizeStart(event) {
|
|
317
|
+
event.preventDefault();
|
|
318
|
+
this.isResizingLeft = true;
|
|
319
|
+
this.resizeStartX = event.clientX;
|
|
320
|
+
this.resizeStartWidth = this.leftSidebarWidth;
|
|
321
|
+
document.addEventListener('mousemove', this.onLeftResizeMove);
|
|
322
|
+
document.addEventListener('mouseup', this.onLeftResizeEnd);
|
|
323
|
+
document.body.style.cursor = 'ew-resize';
|
|
324
|
+
document.body.style.userSelect = 'none';
|
|
325
|
+
}
|
|
326
|
+
// ─── Private ─────────────────────────────────────────────────
|
|
327
|
+
checkForRecovery() {
|
|
328
|
+
if (this.persistenceService.hasLocalStorageData()) {
|
|
329
|
+
const age = this.persistenceService.getLocalStorageAge();
|
|
330
|
+
const message = age
|
|
331
|
+
? `Trovato salvataggio del ${age.toLocaleString('it-IT')}. Vuoi recuperarlo?`
|
|
332
|
+
: 'Trovato salvataggio precedente. Vuoi recuperarlo?';
|
|
333
|
+
const ref = this.snackBar.open(message, 'Recupera', { duration: 10000 });
|
|
334
|
+
ref.onAction().subscribe(() => {
|
|
335
|
+
const schema = this.persistenceService.loadFromLocalStorage();
|
|
336
|
+
if (schema) {
|
|
337
|
+
this.stateService.setSchema(schema);
|
|
338
|
+
this.showMessage('Schema recuperato');
|
|
339
|
+
}
|
|
340
|
+
});
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
autoSave() {
|
|
344
|
+
try {
|
|
345
|
+
this.persistenceService.saveToLocalStorage(this.state.schema);
|
|
346
|
+
this.stateService.markAsSaved();
|
|
347
|
+
}
|
|
348
|
+
catch {
|
|
349
|
+
// Silenzioso per auto-save
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
updateBreadcrumb() {
|
|
353
|
+
this.breadcrumb = [
|
|
354
|
+
{
|
|
355
|
+
label: this.state.schema.title || 'Form',
|
|
356
|
+
id: this.state.schema.id,
|
|
357
|
+
type: 'form',
|
|
358
|
+
active: !this.state.selectedSectionId && !this.state.selectedFieldKey,
|
|
359
|
+
},
|
|
360
|
+
];
|
|
361
|
+
if (this.state.selectedSectionId) {
|
|
362
|
+
const section = this.state.schema.sections.find((s) => s.id === this.state.selectedSectionId);
|
|
363
|
+
if (section) {
|
|
364
|
+
this.breadcrumb.push({
|
|
365
|
+
label: section.title || 'Sezione',
|
|
366
|
+
id: section.id,
|
|
367
|
+
type: 'section',
|
|
368
|
+
active: !this.state.selectedFieldKey,
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
if (this.state.selectedFieldKey) {
|
|
373
|
+
const field = this.stateService.findFieldByKey(this.state.selectedFieldKey);
|
|
374
|
+
if (field) {
|
|
375
|
+
this.breadcrumb.push({
|
|
376
|
+
label: field.label,
|
|
377
|
+
id: field.key,
|
|
378
|
+
type: 'field',
|
|
379
|
+
active: true,
|
|
380
|
+
});
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
updateAvailableFields() {
|
|
385
|
+
const fields = [];
|
|
386
|
+
this.state.schema.sections.forEach((section) => {
|
|
387
|
+
section.fields.forEach((field) => {
|
|
388
|
+
const af = {
|
|
389
|
+
key: field.key,
|
|
390
|
+
label: field.label,
|
|
391
|
+
type: field.type,
|
|
392
|
+
};
|
|
393
|
+
if (field.options && Array.isArray(field.options) && field.options.length > 0) {
|
|
394
|
+
af.options = field.options.map((opt) => ({
|
|
395
|
+
value: opt.value,
|
|
396
|
+
label: opt.label,
|
|
397
|
+
}));
|
|
398
|
+
}
|
|
399
|
+
fields.push(af);
|
|
400
|
+
});
|
|
401
|
+
});
|
|
402
|
+
this.availableFields = fields;
|
|
403
|
+
}
|
|
404
|
+
showMessage(message, isError = false) {
|
|
405
|
+
this.snackBar.open(message, 'Chiudi', {
|
|
406
|
+
duration: 3000,
|
|
407
|
+
panelClass: isError ? ['ui-snackbar-error'] : [],
|
|
408
|
+
});
|
|
409
|
+
}
|
|
410
|
+
loadSidebarWidth() {
|
|
411
|
+
try {
|
|
412
|
+
const savedRight = localStorage.getItem(this.SIDEBAR_WIDTH_KEY);
|
|
413
|
+
if (savedRight) {
|
|
414
|
+
const w = parseInt(savedRight, 10);
|
|
415
|
+
if (!isNaN(w) && w >= this.MIN_SIDEBAR_WIDTH && w <= this.MAX_SIDEBAR_WIDTH) {
|
|
416
|
+
this.rightSidebarWidth = w;
|
|
417
|
+
}
|
|
418
|
+
}
|
|
419
|
+
const savedLeft = localStorage.getItem(this.LEFT_SIDEBAR_WIDTH_KEY);
|
|
420
|
+
if (savedLeft) {
|
|
421
|
+
const w = parseInt(savedLeft, 10);
|
|
422
|
+
if (!isNaN(w) && w >= this.MIN_LEFT_SIDEBAR_WIDTH && w <= this.MAX_LEFT_SIDEBAR_WIDTH) {
|
|
423
|
+
this.leftSidebarWidth = w;
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
catch {
|
|
428
|
+
// Silenzioso
|
|
429
|
+
}
|
|
430
|
+
}
|
|
431
|
+
saveSidebarWidth() {
|
|
432
|
+
try {
|
|
433
|
+
localStorage.setItem(this.SIDEBAR_WIDTH_KEY, this.rightSidebarWidth.toString());
|
|
434
|
+
}
|
|
435
|
+
catch {
|
|
436
|
+
// Silenzioso
|
|
437
|
+
}
|
|
438
|
+
}
|
|
439
|
+
saveLeftSidebarWidth() {
|
|
440
|
+
try {
|
|
441
|
+
localStorage.setItem(this.LEFT_SIDEBAR_WIDTH_KEY, this.leftSidebarWidth.toString());
|
|
442
|
+
}
|
|
443
|
+
catch {
|
|
444
|
+
// Silenzioso
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormBuilderEditorComponent, deps: [{ token: i1.MatDialogRef, optional: true }, { token: MAT_DIALOG_DATA, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
448
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiFormBuilderEditorComponent, isStandalone: true, selector: "ui-form-builder-editor", providers: [UiEditorStateService, UiEditorPersistenceService, UiEditorFieldFactoryService], ngImport: i0, template: `
|
|
449
|
+
<div class="editor-root" [class.dialog-mode]="isDialogMode">
|
|
450
|
+
<!-- Header con chiusura (solo in modale) -->
|
|
451
|
+
@if (isDialogMode) {
|
|
452
|
+
<div class="editor-dialog-header">
|
|
453
|
+
<h2>Form Builder - Editor Visuale</h2>
|
|
454
|
+
<ui-button
|
|
455
|
+
icon="x"
|
|
456
|
+
variant="ghost"
|
|
457
|
+
size="md"
|
|
458
|
+
ariaLabel="Chiudi editor"
|
|
459
|
+
tooltip="Chiudi editor"
|
|
460
|
+
(click)="closeDialog()"
|
|
461
|
+
/>
|
|
462
|
+
</div>
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
<!-- Toolbar -->
|
|
466
|
+
<ui-editor-toolbar
|
|
467
|
+
[breadcrumb]="breadcrumb"
|
|
468
|
+
[isDirty]="state.isDirty"
|
|
469
|
+
[lastSaved]="state.lastSaved"
|
|
470
|
+
(newForm)="onNewForm()"
|
|
471
|
+
(importJson)="onImportJson()"
|
|
472
|
+
(exportJson)="onExportJson()"
|
|
473
|
+
(save)="onSave()"
|
|
474
|
+
(clear)="onClear()"
|
|
475
|
+
(addSection)="onAddSection()"
|
|
476
|
+
/>
|
|
477
|
+
|
|
478
|
+
<!-- Tabs: Editor / Preview -->
|
|
479
|
+
<mat-tab-group
|
|
480
|
+
[selectedIndex]="selectedTabIndex"
|
|
481
|
+
(selectedIndexChange)="onTabChange($event)"
|
|
482
|
+
class="editor-tabs"
|
|
483
|
+
>
|
|
484
|
+
<!-- Tab Editor -->
|
|
485
|
+
<mat-tab label="Editor">
|
|
486
|
+
<div class="editor-layout" cdkDropListGroup>
|
|
487
|
+
<!-- Palette (sidebar sinistra) -->
|
|
488
|
+
<div
|
|
489
|
+
class="editor-sidebar left"
|
|
490
|
+
[style.width.px]="leftSidebarWidth"
|
|
491
|
+
>
|
|
492
|
+
<ui-field-palette />
|
|
493
|
+
<div
|
|
494
|
+
class="resize-handle resize-handle--right"
|
|
495
|
+
[class.resizing]="isResizingLeft"
|
|
496
|
+
(mousedown)="onLeftResizeStart($event)"
|
|
497
|
+
></div>
|
|
498
|
+
</div>
|
|
499
|
+
|
|
500
|
+
<!-- Area centrale con sezioni -->
|
|
501
|
+
<div class="editor-main">
|
|
502
|
+
<div class="sections-container">
|
|
503
|
+
@for (section of state.schema.sections; track section.id) {
|
|
504
|
+
<ui-section-editor
|
|
505
|
+
[section]="section"
|
|
506
|
+
[isSelected]="state.selectedSectionId === section.id"
|
|
507
|
+
[selectedFieldKey]="state.selectedFieldKey"
|
|
508
|
+
(sectionUpdate)="
|
|
509
|
+
stateService.updateSection(section.id, $event)
|
|
510
|
+
"
|
|
511
|
+
(sectionDelete)="onSectionDelete(section.id)"
|
|
512
|
+
(sectionDuplicate)="onSectionDuplicate(section.id)"
|
|
513
|
+
(fieldSelect)="onFieldSelect(section.id, $event)"
|
|
514
|
+
(fieldDuplicate)="onFieldDuplicate(section.id, $event)"
|
|
515
|
+
(fieldDelete)="onFieldDelete(section.id, $event)"
|
|
516
|
+
(fieldDrop)="onFieldDrop($event)"
|
|
517
|
+
/>
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
@if (state.schema.sections.length === 0) {
|
|
521
|
+
<div class="empty-state">
|
|
522
|
+
<lucide-icon
|
|
523
|
+
name="layout-template"
|
|
524
|
+
[size]="64"
|
|
525
|
+
></lucide-icon>
|
|
526
|
+
<h3>Nessuna sezione presente</h3>
|
|
527
|
+
<p>
|
|
528
|
+
Clicca su "Sezione" nella toolbar per aggiungere la prima
|
|
529
|
+
sezione
|
|
530
|
+
</p>
|
|
531
|
+
<ui-button
|
|
532
|
+
label="Aggiungi Prima Sezione"
|
|
533
|
+
variant="primary"
|
|
534
|
+
icon="plus"
|
|
535
|
+
(click)="onAddSection()"
|
|
536
|
+
/>
|
|
537
|
+
</div>
|
|
538
|
+
}
|
|
539
|
+
</div>
|
|
540
|
+
</div>
|
|
541
|
+
|
|
542
|
+
<!-- Config panel (sidebar destra) -->
|
|
543
|
+
<div
|
|
544
|
+
class="editor-sidebar right"
|
|
545
|
+
[style.width.px]="rightSidebarWidth"
|
|
546
|
+
>
|
|
547
|
+
<div
|
|
548
|
+
class="resize-handle"
|
|
549
|
+
[class.resizing]="isResizing"
|
|
550
|
+
(mousedown)="onResizeStart($event)"
|
|
551
|
+
></div>
|
|
552
|
+
<ui-editor-field-config-panel
|
|
553
|
+
[field]="selectedField"
|
|
554
|
+
[availableFields]="availableFields"
|
|
555
|
+
(fieldUpdate)="
|
|
556
|
+
onFieldUpdate(
|
|
557
|
+
selectedFieldSectionId!,
|
|
558
|
+
state.selectedFieldKey!,
|
|
559
|
+
$event
|
|
560
|
+
)
|
|
561
|
+
"
|
|
562
|
+
/>
|
|
563
|
+
</div>
|
|
564
|
+
</div>
|
|
565
|
+
</mat-tab>
|
|
566
|
+
|
|
567
|
+
<!-- Tab Preview -->
|
|
568
|
+
<mat-tab label="Anteprima">
|
|
569
|
+
<ui-editor-preview-container
|
|
570
|
+
[schema]="state.schema"
|
|
571
|
+
[key]="previewKey"
|
|
572
|
+
(refresh)="refreshPreview()"
|
|
573
|
+
/>
|
|
574
|
+
</mat-tab>
|
|
575
|
+
</mat-tab-group>
|
|
576
|
+
</div>
|
|
577
|
+
`, isInline: true, styles: ["@use \"../../../core/tokens/mixins\" as ui;.editor-root{display:flex;flex-direction:column;height:100vh;background:var(--ui-color-bg-subtle);&.dialog-mode{height:100%}}.editor-dialog-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);box-shadow:var(--ui-shadow-sm);h2{margin:0;font-size:var(--ui-font-size-lg);font-weight:500;color:var(--ui-color-text)}}.editor-tabs{flex:1;display:flex;flex-direction:column;overflow:hidden;::ng-deep .mat-mdc-tab-header{background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border)}::ng-deep .mat-mdc-tab .mdc-tab__text-label{color:var(--ui-color-text-secondary);font-weight:500;font-size:var(--ui-font-size-sm);letter-spacing:.02em}::ng-deep .mat-mdc-tab.mdc-tab--active .mdc-tab__text-label{color:var(--ui-color-primary);font-weight:600}::ng-deep .mat-mdc-tab-header .mdc-tab-indicator__content--underline{border-color:var(--ui-color-primary);border-width:3px}::ng-deep .mat-mdc-tab-body-wrapper{flex:1;overflow:hidden}::ng-deep .mat-mdc-tab-body{height:100%}::ng-deep .mat-mdc-tab-body-content{height:100%;overflow:hidden}}.editor-layout{display:flex;height:100%;overflow:hidden}.editor-sidebar{height:100%;overflow-y:auto;background:var(--ui-color-surface);flex-shrink:0;&.left{position:relative;border-right:1px solid var(--ui-color-border);min-width:200px;max-width:500px}&.right{position:relative;border-left:1px solid var(--ui-color-border);min-width:400px;max-width:1200px}}.editor-main{flex:1;min-width:0;height:100%;overflow-y:auto;padding:var(--ui-spacing-4);background:var(--ui-color-bg-subtle)}.sections-container{max-width:1200px;margin:0 auto;display:flex;flex-direction:column;gap:var(--ui-spacing-4)}.resize-handle{position:absolute;top:0;bottom:0;width:6px;cursor:ew-resize;z-index:10;transition:background-color var(--ui-transition-fast);left:0;&:before{content:\"\";position:absolute;left:-3px;top:0;bottom:0;width:12px}&:hover,&.resizing{background-color:color-mix(in srgb,var(--ui-color-primary) 30%,transparent)}&.resizing{background-color:color-mix(in srgb,var(--ui-color-primary) 50%,transparent)}}.resize-handle--right{left:auto;right:0;&:before{left:auto;right:-3px}}.empty-state{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:var(--ui-spacing-12) var(--ui-spacing-8);text-align:center;color:var(--ui-color-text-muted);lucide-icon{opacity:.4}h3{margin:0 0 var(--ui-spacing-2) 0;font-size:var(--ui-font-size-xl);font-weight:400;color:var(--ui-color-text-secondary)}p{margin:0 0 var(--ui-spacing-6) 0;font-size:var(--ui-font-size-md);color:var(--ui-color-text-muted)}}@media (max-width: 1280px){.editor-sidebar.left{width:250px}}@media (max-width: 960px){.editor-layout{flex-direction:column}.editor-sidebar{&.left{display:none}&.right{position:fixed;right:0;top:64px;width:100%!important;max-width:350px;height:calc(100vh - 64px);z-index:1000;box-shadow:var(--ui-shadow-lg);transform:translate(100%);transition:transform var(--ui-transition-normal);.resize-handle{display:none}}}}\n"], dependencies: [{ kind: "ngmodule", type: DragDropModule }, { kind: "directive", type: i2.CdkDropListGroup, selector: "[cdkDropListGroup]", inputs: ["cdkDropListGroupDisabled"], exportAs: ["cdkDropListGroup"] }, { kind: "ngmodule", type: MatSnackBarModule }, { kind: "ngmodule", type: MatTabsModule }, { kind: "component", type: i3.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i3.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor", "aria-label", "aria-labelledby"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i4.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiEditorToolbarComponent, selector: "ui-editor-toolbar", inputs: ["breadcrumb", "isDirty", "lastSaved"], outputs: ["newForm", "importJson", "exportJson", "save", "clear", "addSection"] }, { kind: "component", type: UiFieldPaletteComponent, selector: "ui-field-palette" }, { kind: "component", type: UiSectionEditorComponent, selector: "ui-section-editor", inputs: ["section", "isSelected", "selectedFieldKey"], outputs: ["sectionUpdate", "sectionDelete", "sectionDuplicate", "fieldSelect", "fieldDuplicate", "fieldDelete", "fieldDrop"] }, { kind: "component", type: UiEditorFieldConfigPanelComponent, selector: "ui-editor-field-config-panel", inputs: ["field", "availableFields"], outputs: ["fieldUpdate"] }, { kind: "component", type: UiEditorPreviewContainerComponent, selector: "ui-editor-preview-container", inputs: ["schema", "key"], outputs: ["refresh"] }, { kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
578
|
+
}
|
|
579
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFormBuilderEditorComponent, decorators: [{
|
|
580
|
+
type: Component,
|
|
581
|
+
args: [{ selector: 'ui-form-builder-editor', standalone: true, imports: [
|
|
582
|
+
DragDropModule,
|
|
583
|
+
MatSnackBarModule,
|
|
584
|
+
MatTabsModule,
|
|
585
|
+
LucideAngularModule,
|
|
586
|
+
UiEditorToolbarComponent,
|
|
587
|
+
UiFieldPaletteComponent,
|
|
588
|
+
UiSectionEditorComponent,
|
|
589
|
+
UiEditorFieldConfigPanelComponent,
|
|
590
|
+
UiEditorPreviewContainerComponent,
|
|
591
|
+
UiButtonComponent,
|
|
592
|
+
], providers: [UiEditorStateService, UiEditorPersistenceService, UiEditorFieldFactoryService], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
593
|
+
<div class="editor-root" [class.dialog-mode]="isDialogMode">
|
|
594
|
+
<!-- Header con chiusura (solo in modale) -->
|
|
595
|
+
@if (isDialogMode) {
|
|
596
|
+
<div class="editor-dialog-header">
|
|
597
|
+
<h2>Form Builder - Editor Visuale</h2>
|
|
598
|
+
<ui-button
|
|
599
|
+
icon="x"
|
|
600
|
+
variant="ghost"
|
|
601
|
+
size="md"
|
|
602
|
+
ariaLabel="Chiudi editor"
|
|
603
|
+
tooltip="Chiudi editor"
|
|
604
|
+
(click)="closeDialog()"
|
|
605
|
+
/>
|
|
606
|
+
</div>
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
<!-- Toolbar -->
|
|
610
|
+
<ui-editor-toolbar
|
|
611
|
+
[breadcrumb]="breadcrumb"
|
|
612
|
+
[isDirty]="state.isDirty"
|
|
613
|
+
[lastSaved]="state.lastSaved"
|
|
614
|
+
(newForm)="onNewForm()"
|
|
615
|
+
(importJson)="onImportJson()"
|
|
616
|
+
(exportJson)="onExportJson()"
|
|
617
|
+
(save)="onSave()"
|
|
618
|
+
(clear)="onClear()"
|
|
619
|
+
(addSection)="onAddSection()"
|
|
620
|
+
/>
|
|
621
|
+
|
|
622
|
+
<!-- Tabs: Editor / Preview -->
|
|
623
|
+
<mat-tab-group
|
|
624
|
+
[selectedIndex]="selectedTabIndex"
|
|
625
|
+
(selectedIndexChange)="onTabChange($event)"
|
|
626
|
+
class="editor-tabs"
|
|
627
|
+
>
|
|
628
|
+
<!-- Tab Editor -->
|
|
629
|
+
<mat-tab label="Editor">
|
|
630
|
+
<div class="editor-layout" cdkDropListGroup>
|
|
631
|
+
<!-- Palette (sidebar sinistra) -->
|
|
632
|
+
<div
|
|
633
|
+
class="editor-sidebar left"
|
|
634
|
+
[style.width.px]="leftSidebarWidth"
|
|
635
|
+
>
|
|
636
|
+
<ui-field-palette />
|
|
637
|
+
<div
|
|
638
|
+
class="resize-handle resize-handle--right"
|
|
639
|
+
[class.resizing]="isResizingLeft"
|
|
640
|
+
(mousedown)="onLeftResizeStart($event)"
|
|
641
|
+
></div>
|
|
642
|
+
</div>
|
|
643
|
+
|
|
644
|
+
<!-- Area centrale con sezioni -->
|
|
645
|
+
<div class="editor-main">
|
|
646
|
+
<div class="sections-container">
|
|
647
|
+
@for (section of state.schema.sections; track section.id) {
|
|
648
|
+
<ui-section-editor
|
|
649
|
+
[section]="section"
|
|
650
|
+
[isSelected]="state.selectedSectionId === section.id"
|
|
651
|
+
[selectedFieldKey]="state.selectedFieldKey"
|
|
652
|
+
(sectionUpdate)="
|
|
653
|
+
stateService.updateSection(section.id, $event)
|
|
654
|
+
"
|
|
655
|
+
(sectionDelete)="onSectionDelete(section.id)"
|
|
656
|
+
(sectionDuplicate)="onSectionDuplicate(section.id)"
|
|
657
|
+
(fieldSelect)="onFieldSelect(section.id, $event)"
|
|
658
|
+
(fieldDuplicate)="onFieldDuplicate(section.id, $event)"
|
|
659
|
+
(fieldDelete)="onFieldDelete(section.id, $event)"
|
|
660
|
+
(fieldDrop)="onFieldDrop($event)"
|
|
661
|
+
/>
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
@if (state.schema.sections.length === 0) {
|
|
665
|
+
<div class="empty-state">
|
|
666
|
+
<lucide-icon
|
|
667
|
+
name="layout-template"
|
|
668
|
+
[size]="64"
|
|
669
|
+
></lucide-icon>
|
|
670
|
+
<h3>Nessuna sezione presente</h3>
|
|
671
|
+
<p>
|
|
672
|
+
Clicca su "Sezione" nella toolbar per aggiungere la prima
|
|
673
|
+
sezione
|
|
674
|
+
</p>
|
|
675
|
+
<ui-button
|
|
676
|
+
label="Aggiungi Prima Sezione"
|
|
677
|
+
variant="primary"
|
|
678
|
+
icon="plus"
|
|
679
|
+
(click)="onAddSection()"
|
|
680
|
+
/>
|
|
681
|
+
</div>
|
|
682
|
+
}
|
|
683
|
+
</div>
|
|
684
|
+
</div>
|
|
685
|
+
|
|
686
|
+
<!-- Config panel (sidebar destra) -->
|
|
687
|
+
<div
|
|
688
|
+
class="editor-sidebar right"
|
|
689
|
+
[style.width.px]="rightSidebarWidth"
|
|
690
|
+
>
|
|
691
|
+
<div
|
|
692
|
+
class="resize-handle"
|
|
693
|
+
[class.resizing]="isResizing"
|
|
694
|
+
(mousedown)="onResizeStart($event)"
|
|
695
|
+
></div>
|
|
696
|
+
<ui-editor-field-config-panel
|
|
697
|
+
[field]="selectedField"
|
|
698
|
+
[availableFields]="availableFields"
|
|
699
|
+
(fieldUpdate)="
|
|
700
|
+
onFieldUpdate(
|
|
701
|
+
selectedFieldSectionId!,
|
|
702
|
+
state.selectedFieldKey!,
|
|
703
|
+
$event
|
|
704
|
+
)
|
|
705
|
+
"
|
|
706
|
+
/>
|
|
707
|
+
</div>
|
|
708
|
+
</div>
|
|
709
|
+
</mat-tab>
|
|
710
|
+
|
|
711
|
+
<!-- Tab Preview -->
|
|
712
|
+
<mat-tab label="Anteprima">
|
|
713
|
+
<ui-editor-preview-container
|
|
714
|
+
[schema]="state.schema"
|
|
715
|
+
[key]="previewKey"
|
|
716
|
+
(refresh)="refreshPreview()"
|
|
717
|
+
/>
|
|
718
|
+
</mat-tab>
|
|
719
|
+
</mat-tab-group>
|
|
720
|
+
</div>
|
|
721
|
+
`, styles: ["@use \"../../../core/tokens/mixins\" as ui;.editor-root{display:flex;flex-direction:column;height:100vh;background:var(--ui-color-bg-subtle);&.dialog-mode{height:100%}}.editor-dialog-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);box-shadow:var(--ui-shadow-sm);h2{margin:0;font-size:var(--ui-font-size-lg);font-weight:500;color:var(--ui-color-text)}}.editor-tabs{flex:1;display:flex;flex-direction:column;overflow:hidden;::ng-deep .mat-mdc-tab-header{background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border)}::ng-deep .mat-mdc-tab .mdc-tab__text-label{color:var(--ui-color-text-secondary);font-weight:500;font-size:var(--ui-font-size-sm);letter-spacing:.02em}::ng-deep .mat-mdc-tab.mdc-tab--active .mdc-tab__text-label{color:var(--ui-color-primary);font-weight:600}::ng-deep .mat-mdc-tab-header .mdc-tab-indicator__content--underline{border-color:var(--ui-color-primary);border-width:3px}::ng-deep .mat-mdc-tab-body-wrapper{flex:1;overflow:hidden}::ng-deep .mat-mdc-tab-body{height:100%}::ng-deep .mat-mdc-tab-body-content{height:100%;overflow:hidden}}.editor-layout{display:flex;height:100%;overflow:hidden}.editor-sidebar{height:100%;overflow-y:auto;background:var(--ui-color-surface);flex-shrink:0;&.left{position:relative;border-right:1px solid var(--ui-color-border);min-width:200px;max-width:500px}&.right{position:relative;border-left:1px solid var(--ui-color-border);min-width:400px;max-width:1200px}}.editor-main{flex:1;min-width:0;height:100%;overflow-y:auto;padding:var(--ui-spacing-4);background:var(--ui-color-bg-subtle)}.sections-container{max-width:1200px;margin:0 auto;display:flex;flex-direction:column;gap:var(--ui-spacing-4)}.resize-handle{position:absolute;top:0;bottom:0;width:6px;cursor:ew-resize;z-index:10;transition:background-color var(--ui-transition-fast);left:0;&:before{content:\"\";position:absolute;left:-3px;top:0;bottom:0;width:12px}&:hover,&.resizing{background-color:color-mix(in srgb,var(--ui-color-primary) 30%,transparent)}&.resizing{background-color:color-mix(in srgb,var(--ui-color-primary) 50%,transparent)}}.resize-handle--right{left:auto;right:0;&:before{left:auto;right:-3px}}.empty-state{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:var(--ui-spacing-12) var(--ui-spacing-8);text-align:center;color:var(--ui-color-text-muted);lucide-icon{opacity:.4}h3{margin:0 0 var(--ui-spacing-2) 0;font-size:var(--ui-font-size-xl);font-weight:400;color:var(--ui-color-text-secondary)}p{margin:0 0 var(--ui-spacing-6) 0;font-size:var(--ui-font-size-md);color:var(--ui-color-text-muted)}}@media (max-width: 1280px){.editor-sidebar.left{width:250px}}@media (max-width: 960px){.editor-layout{flex-direction:column}.editor-sidebar{&.left{display:none}&.right{position:fixed;right:0;top:64px;width:100%!important;max-width:350px;height:calc(100vh - 64px);z-index:1000;box-shadow:var(--ui-shadow-lg);transform:translate(100%);transition:transform var(--ui-transition-normal);.resize-handle{display:none}}}}\n"] }]
|
|
722
|
+
}], ctorParameters: () => [{ type: i1.MatDialogRef, decorators: [{
|
|
723
|
+
type: Optional
|
|
724
|
+
}] }, { type: undefined, decorators: [{
|
|
725
|
+
type: Optional
|
|
726
|
+
}, {
|
|
727
|
+
type: Inject,
|
|
728
|
+
args: [MAT_DIALOG_DATA]
|
|
729
|
+
}] }] });
|
|
730
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-builder-editor.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/form-builder-editor.component.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;GAgBG;AAEH,OAAO,EACL,SAAS,EACT,uBAAuB,EACvB,iBAAiB,EAGjB,MAAM,EACN,QAAQ,EACR,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAe,cAAc,EAAE,MAAM,wBAAwB,CAAC;AACrE,OAAO,EAAgB,eAAe,EAAE,MAAM,0BAA0B,CAAC;AACzE,OAAO,EAAE,WAAW,EAAE,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,MAAM,CAAC;AACzC,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAY3C,OAAO,EAAE,oBAAoB,EAAE,MAAM,iCAAiC,CAAC;AACvE,OAAO,EAAE,0BAA0B,EAAE,MAAM,uCAAuC,CAAC;AACnF,OAAO,EAAE,2BAA2B,EAAE,MAAM,kCAAkC,CAAC;AAE/E,mBAAmB;AACnB,OAAO,EAAE,wBAAwB,EAAE,MAAM,0DAA0D,CAAC;AACpG,OAAO,EAAE,uBAAuB,EAAE,MAAM,wDAAwD,CAAC;AACjG,OAAO,EAAE,wBAAwB,EAAE,MAAM,0DAA0D,CAAC;AACpG,OAAO,EAAE,iCAAiC,EAAE,MAAM,kEAAkE,CAAC;AACrH,OAAO,EAAE,iCAAiC,EAAE,MAAM,gEAAgE,CAAC;AACnH,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;;;;;;AAqYpD,MAAM,OAAO,4BAA4B;IA8DvC,YACqB,SAAqD,EAC5B,UAA8B;QADvD,cAAS,GAAT,SAAS,CAA4C;QAC5B,eAAU,GAAV,UAAU,CAAoB;QA/D5E,mDAAmD;QAC1C,iBAAY,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAErD,6CAA6C;QAC5B,uBAAkB,GAAG,MAAM,CAAC,0BAA0B,CAAC,CAAC;QAEzE,qCAAqC;QACpB,iBAAY,GAAG,MAAM,CAAC,2BAA2B,CAAC,CAAC;QAEpE,0CAA0C;QACzB,aAAQ,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAEhD,iCAAiC;QAChB,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAEjD,4CAA4C;QAC3B,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAKhD,wDAAwD;QACxD,qBAAgB,GAAG,CAAC,CAAC;QAErB,iCAAiC;QACjC,eAAU,GAA6B,EAAE,CAAC;QAE1C,oDAAoD;QACpD,iBAAY,GAAG,KAAK,CAAC;QAErB,mDAAmD;QACnD,eAAU,GAAG,CAAC,CAAC;QAEf,mDAAmD;QACnD,oBAAe,GAA6B,EAAE,CAAC;QAE/C,6CAA6C;QAC7C,qBAAgB,GAAG,GAAG,CAAC;QAEvB,2CAA2C;QAC3C,sBAAiB,GAAG,GAAG,CAAC;QAExB,oDAAoD;QACpD,eAAU,GAAG,KAAK,CAAC;QAEnB,sDAAsD;QACtD,mBAAc,GAAG,KAAK,CAAC;QAEvB,iDAAiD;QAChC,sBAAiB,GAAG,GAAG,CAAC;QACxB,sBAAiB,GAAG,IAAI,CAAC;QAE1C,mDAAmD;QAClC,2BAAsB,GAAG,GAAG,CAAC;QAC7B,2BAAsB,GAAG,GAAG,CAAC;QAEtC,iBAAY,GAAG,CAAC,CAAC;QACjB,qBAAgB,GAAG,CAAC,CAAC;QACZ,sBAAiB,GAAG,+BAA+B,CAAC;QACpD,2BAAsB,GAAG,8BAA8B,CAAC;QAoOjE,iBAAY,GAAG,CAAC,KAAiB,EAAQ,EAAE;YACjD,IAAI,CAAC,IAAI,CAAC,UAAU;gBAAE,OAAO;YAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC;YAChD,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,GAAG,CAC/B,IAAI,CAAC,iBAAiB,EACtB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,CAChE,CAAC;YACF,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CAAC;QAEM,gBAAW,GAAG,GAAS,EAAE;YAC/B,IAAI,CAAC,IAAI,CAAC,UAAU;gBAAE,OAAO;YAC7B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;YACxB,QAAQ,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YAC7D,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;YAC1D,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,EAAE,CAAC;YAChC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,EAAE,CAAC;YACpC,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC1B,CAAC,CAAC;QAgBM,qBAAgB,GAAG,CAAC,KAAiB,EAAQ,EAAE;YACrD,IAAI,CAAC,IAAI,CAAC,cAAc;gBAAE,OAAO;YACjC,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC;YAChD,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,GAAG,CAC9B,IAAI,CAAC,sBAAsB,EAC3B,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,sBAAsB,EAAE,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,CACrE,CAAC;YACF,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CAAC;QAEM,oBAAe,GAAG,GAAS,EAAE;YACnC,IAAI,CAAC,IAAI,CAAC,cAAc;gBAAE,OAAO;YACjC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;YAC5B,QAAQ,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;YACjE,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC;YAC9D,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,EAAE,CAAC;YAChC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,EAAE,CAAC;YACpC,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9B,CAAC,CAAC;QAlRA,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,CAAC;QACjD,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;QAErC,oDAAoD;QACpD,IAAI,IAAI,CAAC,UAAU,EAAE,aAAa,EAAE,CAAC;YACnC,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAC7D,CAAC;QAED,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,QAAQ;QACN,4BAA4B;QAC5B,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YAC1E,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;YACnB,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACxB,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,gDAAgD;QAChD,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAExB,4BAA4B;QAC5B,QAAQ,CAAC,KAAK,CAAC;aACZ,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,GAAG,EAAE;YACd,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;gBACvB,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClB,CAAC;QACH,CAAC,CAAC,CAAC;IACP,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAED,gEAAgE;IAEhE,kCAAkC;IAClC,YAAY;QACV,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;QAC/B,IAAI,CAAC,WAAW,CAAC,kBAAkB,CAAC,CAAC;IACvC,CAAC;IAED,8BAA8B;IAC9B,MAAM;QACJ,IAAI,CAAC;YACH,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAC9D,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;YAChC,IAAI,CAAC,WAAW,CAAC,sBAAsB,CAAC,CAAC;QAC3C,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,OAAO,IAAI,+BAA+B,EAAE,IAAI,CAAC,CAAC;QAC3E,CAAC;IACH,CAAC;IAED,0BAA0B;IAC1B,SAAS;QACP,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,4CAA4C,CAAC,EAAE,CAAC;YACjF,OAAO;QACT,CAAC;QACD,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;QAC1B,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;IACxC,CAAC;IAED,4BAA4B;IAC5B,YAAY;QACV,MAAM,KAAK,GAAG,QAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAC9C,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC;QACpB,KAAK,CAAC,MAAM,GAAG,OAAO,CAAC;QAEvB,KAAK,CAAC,QAAQ,GAAG,KAAK,EAAE,CAAM,EAAE,EAAE;YAChC,MAAM,IAAI,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC/B,IAAI,CAAC,IAAI;gBAAE,OAAO;YAElB,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;gBAClE,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,+CAA+C,CAAC,EAAE,CAAC;oBACpF,OAAO;gBACT,CAAC;gBACD,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;gBACpC,IAAI,CAAC,WAAW,CAAC,+BAA+B,CAAC,CAAC;YACpD,CAAC;YAAC,OAAO,KAAU,EAAE,CAAC;gBACpB,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,OAAO,IAAI,yBAAyB,EAAE,IAAI,CAAC,CAAC;YACrE,CAAC;QACH,CAAC,CAAC;QAEF,KAAK,CAAC,KAAK,EAAE,CAAC;IAChB,CAAC;IAED,8BAA8B;IAC9B,YAAY;QACV,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,EAAE,CAAC;YACtD,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;gBACtB,MAAM,OAAO,GAAG,uBAAuB,UAAU,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC5F,IAAI,CAAC,OAAO,CAAC,GAAG,OAAO,yBAAyB,CAAC;oBAAE,OAAO;YAC5D,CAAC;YACD,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YACxD,IAAI,CAAC,WAAW,CAAC,+BAA+B,CAAC,CAAC;QACpD,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,OAAO,IAAI,yBAAyB,EAAE,IAAI,CAAC,CAAC;QACrE,CAAC;IACH,CAAC;IAED,+BAA+B;IAC/B,OAAO;QACL,IAAI,CAAC,OAAO,CAAC,uEAAuE,CAAC;YAAE,OAAO;QAC9F,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACxD,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;QAChC,IAAI,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC;IACpC,CAAC;IAED,gEAAgE;IAEhE,kBAAkB,CAAC,SAAiB;QAClC,IAAI,CAAC,YAAY,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC;QAC9C,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;IACxC,CAAC;IAED,eAAe,CAAC,SAAiB;QAC/B,IAAI,CAAC,OAAO,CAAC,wCAAwC,CAAC;YAAE,OAAO;QAC/D,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;QAC3C,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;IACxC,CAAC;IAED,gEAAgE;IAEhE,aAAa,CAAC,SAAiB,EAAE,QAAgB;QAC/C,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;IACrD,CAAC;IAED,gBAAgB,CAAC,SAAiB,EAAE,QAAgB;QAClD,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;QACtD,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,CAAC;IACtC,CAAC;IAED,aAAa,CAAC,SAAiB,EAAE,QAAgB;QAC/C,IAAI,CAAC,OAAO,CAAC,sCAAsC,CAAC;YAAE,OAAO;QAC7D,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;QACnD,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,CAAC;IACtC,CAAC;IAED,aAAa,CAAC,SAAiB,EAAE,QAAgB,EAAE,OAAuC;QACxF,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,SAAS,EAAE,QAAQ,EAAE,OAAO,CAAC,CAAC;IAC9D,CAAC;IAED,kEAAkE;IAClE,WAAW,CAAC,KAAuB;QACjC,MAAM,MAAM,GAAG,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC;QAE1C,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAClC,qBAAqB;YACrB,MAAM,WAAW,GAA6B,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC;YAC9D,MAAM,OAAO,GAAkB,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC;YACpD,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;YACxE,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,OAAO,CAAC,EAAE,EAAE,QAAQ,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;YAC5E,IAAI,CAAC,WAAW,CAAC,SAAS,WAAW,CAAC,KAAK,WAAW,CAAC,CAAC;QAC1D,CAAC;aAAM,IAAI,KAAK,CAAC,iBAAiB,KAAK,KAAK,CAAC,SAAS,EAAE,CAAC;YACvD,0BAA0B;YAC1B,MAAM,KAAK,GAA0B,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC;YACrD,MAAM,WAAW,GAAkB,KAAK,CAAC,iBAAiB,CAAC,IAAI,CAAC;YAChE,MAAM,SAAS,GAAkB,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC;YACtD,IAAI,CAAC,YAAY,CAAC,wBAAwB,CAAC,WAAW,CAAC,EAAE,EAAE,SAAS,CAAC,EAAE,EAAE,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;YACxG,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,gCAAgC;YAChC,MAAM,OAAO,GAAkB,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC;YACpD,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,OAAO,CAAC,EAAE,EAAE,KAAK,CAAC,aAAa,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;QAC5F,CAAC;QAED,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,gEAAgE;IAEhE,qCAAqC;IACrC,IAAI,aAAa;QACf,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB;YAAE,OAAO,IAAI,CAAC;QAC9C,OAAO,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;IACvE,CAAC;IAED,8CAA8C;IAC9C,IAAI,sBAAsB;QACxB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB;YAAE,OAAO,IAAI,CAAC;QAC9C,OAAO,IAAI,CAAC,YAAY,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,EAAE,EAAE,IAAI,IAAI,CAAC;IAC1F,CAAC;IAED,gEAAgE;IAEhE,WAAW,CAAC,KAAa;QACvB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAC9B,IAAI,KAAK,KAAK,CAAC;YAAE,IAAI,CAAC,cAAc,EAAE,CAAC;IACzC,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,UAAU,EAAE,CAAC;QAClB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,gEAAgE;IAEhE,uDAAuD;IACvD,WAAW;QACT,IAAI,CAAC,SAAS,EAAE,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAC3C,CAAC;IAED,gEAAgE;IAEhE,aAAa,CAAC,KAAiB;QAC7B,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC;QAClC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,iBAAiB,CAAC;QAE/C,QAAQ,CAAC,gBAAgB,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;QAC1D,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;QACvD,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,WAAW,CAAC;QACzC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,MAAM,CAAC;IAC1C,CAAC;IAsBD,gEAAgE;IAEhE,iBAAiB,CAAC,KAAiB;QACjC,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC3B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC;QAClC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAE9C,QAAQ,CAAC,gBAAgB,CAAC,WAAW,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC9D,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC;QAC3D,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,WAAW,CAAC;QACzC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,MAAM,CAAC;IAC1C,CAAC;IAsBD,gEAAgE;IAExD,gBAAgB;QACtB,IAAI,IAAI,CAAC,kBAAkB,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAClD,MAAM,GAAG,GAAG,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,EAAE,CAAC;YACzD,MAAM,OAAO,GAAG,GAAG;gBACjB,CAAC,CAAC,2BAA2B,GAAG,CAAC,cAAc,CAAC,OAAO,CAAC,qBAAqB;gBAC7E,CAAC,CAAC,mDAAmD,CAAC;YAExD,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,CAAC;YACzE,GAAG,CAAC,QAAQ,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE;gBAC5B,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,EAAE,CAAC;gBAC9D,IAAI,MAAM,EAAE,CAAC;oBACX,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;oBACpC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;gBACxC,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,QAAQ;QACd,IAAI,CAAC;YACH,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAC9D,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;QAClC,CAAC;QAAC,MAAM,CAAC;YACP,2BAA2B;QAC7B,CAAC;IACH,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC,UAAU,GAAG;YAChB;gBACE,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,MAAM;gBACxC,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE;gBACxB,IAAI,EAAE,MAAM;gBACZ,MAAM,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB;aACtE;SACF,CAAC;QAEF,IAAI,IAAI,CAAC,KAAK,CAAC,iBAAiB,EAAE,CAAC;YACjC,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;YAC9F,IAAI,OAAO,EAAE,CAAC;gBACZ,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;oBACnB,KAAK,EAAE,OAAO,CAAC,KAAK,IAAI,SAAS;oBACjC,EAAE,EAAE,OAAO,CAAC,EAAE;oBACd,IAAI,EAAE,SAAS;oBACf,MAAM,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB;iBACrC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAChC,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;YAC5E,IAAI,KAAK,EAAE,CAAC;gBACV,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;oBACnB,KAAK,EAAE,KAAK,CAAC,KAAK;oBAClB,EAAE,EAAE,KAAK,CAAC,GAAG;oBACb,IAAI,EAAE,OAAO;oBACb,MAAM,EAAE,IAAI;iBACb,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAEO,qBAAqB;QAC3B,MAAM,MAAM,GAA6B,EAAE,CAAC;QAC5C,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;gBAC/B,MAAM,EAAE,GAA2B;oBACjC,GAAG,EAAE,KAAK,CAAC,GAAG;oBACd,KAAK,EAAE,KAAK,CAAC,KAAK;oBAClB,IAAI,EAAE,KAAK,CAAC,IAAI;iBACjB,CAAC;gBACF,IAAI,KAAK,CAAC,OAAO,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC9E,EAAE,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;wBACvC,KAAK,EAAE,GAAG,CAAC,KAAK;wBAChB,KAAK,EAAE,GAAG,CAAC,KAAK;qBACjB,CAAC,CAAC,CAAC;gBACN,CAAC;gBACD,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAClB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;IAChC,CAAC;IAEO,WAAW,CAAC,OAAe,EAAE,OAAO,GAAG,KAAK;QAClD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,EAAE;YACpC,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,EAAE;SACjD,CAAC,CAAC;IACL,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAChE,IAAI,UAAU,EAAE,CAAC;gBACf,MAAM,CAAC,GAAG,QAAQ,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;gBACnC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,iBAAiB,IAAI,CAAC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;oBAC5E,IAAI,CAAC,iBAAiB,GAAG,CAAC,CAAC;gBAC7B,CAAC;YACH,CAAC;YACD,MAAM,SAAS,GAAG,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;YACpE,IAAI,SAAS,EAAE,CAAC;gBACd,MAAM,CAAC,GAAG,QAAQ,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;gBAClC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,sBAAsB,IAAI,CAAC,IAAI,IAAI,CAAC,sBAAsB,EAAE,CAAC;oBACtF,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;gBAC5B,CAAC;YACH,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,aAAa;QACf,CAAC;IACH,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC;YACH,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,iBAAiB,CAAC,QAAQ,EAAE,CAAC,CAAC;QAClF,CAAC;QAAC,MAAM,CAAC;YACP,aAAa;QACf,CAAC;IACH,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC;YACH,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,sBAAsB,EAAE,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,CAAC,CAAC;QACtF,CAAC;QAAC,MAAM,CAAC;YACP,aAAa;QACf,CAAC;IACH,CAAC;+GArdU,4BAA4B,8DAgEjB,eAAe;mGAhE1B,4BAA4B,qEApX5B,CAAC,oBAAoB,EAAE,0BAA0B,EAAE,2BAA2B,CAAC,0BAEhF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiIT,upGA9IC,cAAc,sLACd,iBAAiB,8BACjB,aAAa,4oBACb,mBAAmB,gPACnB,wBAAwB,+LACxB,uBAAuB,6DACvB,wBAAwB,mPACxB,iCAAiC,yIACjC,iCAAiC,yHACjC,iBAAiB;;4FAsXR,4BAA4B;kBAnYxC,SAAS;+BACE,wBAAwB,cACtB,IAAI,WACP;wBACP,cAAc;wBACd,iBAAiB;wBACjB,aAAa;wBACb,mBAAmB;wBACnB,wBAAwB;wBACxB,uBAAuB;wBACvB,wBAAwB;wBACxB,iCAAiC;wBACjC,iCAAiC;wBACjC,iBAAiB;qBAClB,aACU,CAAC,oBAAoB,EAAE,0BAA0B,EAAE,2BAA2B,CAAC,mBACzE,uBAAuB,CAAC,MAAM,YACrC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiIT;;0BAgTE,QAAQ;;0BACR,QAAQ;;0BAAI,MAAM;2BAAC,eAAe","sourcesContent":["/**\r\n * @module ng-ui-system/form-builder-editor\r\n * Componente principale dell'editor visuale del form builder.\r\n *\r\n * Layout a tre pannelli:\r\n * - Sinistra: palette dei tipi di campo (drag source)\r\n * - Centro: sezioni con campi (drop target + editor inline)\r\n * - Destra: pannello di configurazione del campo selezionato\r\n *\r\n * Funzionalita principali:\r\n * - Drag & drop dei campi dalla palette alle sezioni\r\n * - Riordino campi all'interno e tra sezioni\r\n * - Configurazione campi in tempo reale\r\n * - Preview live del form\r\n * - Import/export JSON\r\n * - Auto-save su localStorage\r\n */\r\n\r\nimport {\r\n  Component,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  OnInit,\r\n  OnDestroy,\r\n  inject,\r\n  Optional,\r\n  Inject,\r\n} from '@angular/core';\r\nimport { CdkDragDrop, DragDropModule } from '@angular/cdk/drag-drop';\r\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\r\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\r\nimport { MatTabsModule } from '@angular/material/tabs';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subject, interval } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\n\r\nimport { UiFormSchema, UiFormSection, UiFormFieldDescriptor } from '../form-builder/types/index';\r\n\r\nimport {\r\n  UiEditorState,\r\n  UiEditorBreadcrumbItem,\r\n  UiEditorFieldPaletteItem,\r\n  UiEditorAvailableField,\r\n  UiEditorDialogData,\r\n} from './types/editor.types';\r\n\r\nimport { UiEditorStateService } from './services/editor-state.service';\r\nimport { UiEditorPersistenceService } from './services/editor-persistence.service';\r\nimport { UiEditorFieldFactoryService } from './services/field-factory.service';\r\n\r\n// Sotto-componenti\r\nimport { UiEditorToolbarComponent } from './sub-components/editor-toolbar/editor-toolbar.component';\r\nimport { UiFieldPaletteComponent } from './sub-components/field-palette/field-palette.component';\r\nimport { UiSectionEditorComponent } from './sub-components/section-editor/section-editor.component';\r\nimport { UiEditorFieldConfigPanelComponent } from './sub-components/field-config-panel/field-config-panel.component';\r\nimport { UiEditorPreviewContainerComponent } from './sub-components/preview-container/preview-container.component';\r\nimport { UiButtonComponent } from '../button/index';\r\n\r\n@Component({\r\n  selector: 'ui-form-builder-editor',\r\n  standalone: true,\r\n  imports: [\r\n    DragDropModule,\r\n    MatSnackBarModule,\r\n    MatTabsModule,\r\n    LucideAngularModule,\r\n    UiEditorToolbarComponent,\r\n    UiFieldPaletteComponent,\r\n    UiSectionEditorComponent,\r\n    UiEditorFieldConfigPanelComponent,\r\n    UiEditorPreviewContainerComponent,\r\n    UiButtonComponent,\r\n  ],\r\n  providers: [UiEditorStateService, UiEditorPersistenceService, UiEditorFieldFactoryService],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"editor-root\" [class.dialog-mode]=\"isDialogMode\">\r\n      <!-- Header con chiusura (solo in modale) -->\r\n      @if (isDialogMode) {\r\n        <div class=\"editor-dialog-header\">\r\n          <h2>Form Builder - Editor Visuale</h2>\r\n          <ui-button\r\n            icon=\"x\"\r\n            variant=\"ghost\"\r\n            size=\"md\"\r\n            ariaLabel=\"Chiudi editor\"\r\n            tooltip=\"Chiudi editor\"\r\n            (click)=\"closeDialog()\"\r\n          />\r\n        </div>\r\n      }\r\n\r\n      <!-- Toolbar -->\r\n      <ui-editor-toolbar\r\n        [breadcrumb]=\"breadcrumb\"\r\n        [isDirty]=\"state.isDirty\"\r\n        [lastSaved]=\"state.lastSaved\"\r\n        (newForm)=\"onNewForm()\"\r\n        (importJson)=\"onImportJson()\"\r\n        (exportJson)=\"onExportJson()\"\r\n        (save)=\"onSave()\"\r\n        (clear)=\"onClear()\"\r\n        (addSection)=\"onAddSection()\"\r\n      />\r\n\r\n      <!-- Tabs: Editor / Preview -->\r\n      <mat-tab-group\r\n        [selectedIndex]=\"selectedTabIndex\"\r\n        (selectedIndexChange)=\"onTabChange($event)\"\r\n        class=\"editor-tabs\"\r\n      >\r\n        <!-- Tab Editor -->\r\n        <mat-tab label=\"Editor\">\r\n          <div class=\"editor-layout\" cdkDropListGroup>\r\n            <!-- Palette (sidebar sinistra) -->\r\n            <div\r\n              class=\"editor-sidebar left\"\r\n              [style.width.px]=\"leftSidebarWidth\"\r\n            >\r\n              <ui-field-palette />\r\n              <div\r\n                class=\"resize-handle resize-handle--right\"\r\n                [class.resizing]=\"isResizingLeft\"\r\n                (mousedown)=\"onLeftResizeStart($event)\"\r\n              ></div>\r\n            </div>\r\n\r\n            <!-- Area centrale con sezioni -->\r\n            <div class=\"editor-main\">\r\n              <div class=\"sections-container\">\r\n                @for (section of state.schema.sections; track section.id) {\r\n                  <ui-section-editor\r\n                    [section]=\"section\"\r\n                    [isSelected]=\"state.selectedSectionId === section.id\"\r\n                    [selectedFieldKey]=\"state.selectedFieldKey\"\r\n                    (sectionUpdate)=\"\r\n                      stateService.updateSection(section.id, $event)\r\n                    \"\r\n                    (sectionDelete)=\"onSectionDelete(section.id)\"\r\n                    (sectionDuplicate)=\"onSectionDuplicate(section.id)\"\r\n                    (fieldSelect)=\"onFieldSelect(section.id, $event)\"\r\n                    (fieldDuplicate)=\"onFieldDuplicate(section.id, $event)\"\r\n                    (fieldDelete)=\"onFieldDelete(section.id, $event)\"\r\n                    (fieldDrop)=\"onFieldDrop($event)\"\r\n                  />\r\n                }\r\n\r\n                @if (state.schema.sections.length === 0) {\r\n                  <div class=\"empty-state\">\r\n                    <lucide-icon\r\n                      name=\"layout-template\"\r\n                      [size]=\"64\"\r\n                    ></lucide-icon>\r\n                    <h3>Nessuna sezione presente</h3>\r\n                    <p>\r\n                      Clicca su \"Sezione\" nella toolbar per aggiungere la prima\r\n                      sezione\r\n                    </p>\r\n                    <ui-button\r\n                      label=\"Aggiungi Prima Sezione\"\r\n                      variant=\"primary\"\r\n                      icon=\"plus\"\r\n                      (click)=\"onAddSection()\"\r\n                    />\r\n                  </div>\r\n                }\r\n              </div>\r\n            </div>\r\n\r\n            <!-- Config panel (sidebar destra) -->\r\n            <div\r\n              class=\"editor-sidebar right\"\r\n              [style.width.px]=\"rightSidebarWidth\"\r\n            >\r\n              <div\r\n                class=\"resize-handle\"\r\n                [class.resizing]=\"isResizing\"\r\n                (mousedown)=\"onResizeStart($event)\"\r\n              ></div>\r\n              <ui-editor-field-config-panel\r\n                [field]=\"selectedField\"\r\n                [availableFields]=\"availableFields\"\r\n                (fieldUpdate)=\"\r\n                  onFieldUpdate(\r\n                    selectedFieldSectionId!,\r\n                    state.selectedFieldKey!,\r\n                    $event\r\n                  )\r\n                \"\r\n              />\r\n            </div>\r\n          </div>\r\n        </mat-tab>\r\n\r\n        <!-- Tab Preview -->\r\n        <mat-tab label=\"Anteprima\">\r\n          <ui-editor-preview-container\r\n            [schema]=\"state.schema\"\r\n            [key]=\"previewKey\"\r\n            (refresh)=\"refreshPreview()\"\r\n          />\r\n        </mat-tab>\r\n      </mat-tab-group>\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n      @use '../../../core/tokens/mixins' as ui;\r\n\r\n      .editor-root {\r\n        display: flex;\r\n        flex-direction: column;\r\n        height: 100vh;\r\n        background: var(--ui-color-bg-subtle);\r\n\r\n        &.dialog-mode {\r\n          height: 100%;\r\n        }\r\n      }\r\n\r\n      .editor-dialog-header {\r\n        display: flex;\r\n        align-items: center;\r\n        justify-content: space-between;\r\n        padding: var(--ui-spacing-3) var(--ui-spacing-4);\r\n        background: var(--ui-color-surface);\r\n        border-bottom: 1px solid var(--ui-color-border);\r\n        box-shadow: var(--ui-shadow-sm);\r\n\r\n        h2 {\r\n          margin: 0;\r\n          font-size: var(--ui-font-size-lg);\r\n          font-weight: 500;\r\n          color: var(--ui-color-text);\r\n        }\r\n\r\n        /* Stile per il pulsante di chiusura (gestito da UiButton) */\r\n      }\r\n\r\n      .editor-tabs {\r\n        flex: 1;\r\n        display: flex;\r\n        flex-direction: column;\r\n        overflow: hidden;\r\n\r\n        /* Stile intestazione tab */\r\n        ::ng-deep .mat-mdc-tab-header {\r\n          background: var(--ui-color-surface);\r\n          border-bottom: 1px solid var(--ui-color-border);\r\n        }\r\n\r\n        /* Stile etichetta tab */\r\n        ::ng-deep .mat-mdc-tab .mdc-tab__text-label {\r\n          color: var(--ui-color-text-secondary);\r\n          font-weight: 500;\r\n          font-size: var(--ui-font-size-sm);\r\n          letter-spacing: 0.02em;\r\n        }\r\n\r\n        /* Tab attivo: testo evidenziato */\r\n        ::ng-deep .mat-mdc-tab.mdc-tab--active .mdc-tab__text-label {\r\n          color: var(--ui-color-primary);\r\n          font-weight: 600;\r\n        }\r\n\r\n        /* Indicatore tab attivo (ink bar) */\r\n        ::ng-deep .mat-mdc-tab-header .mdc-tab-indicator__content--underline {\r\n          border-color: var(--ui-color-primary);\r\n          border-width: 3px;\r\n        }\r\n\r\n        ::ng-deep .mat-mdc-tab-body-wrapper {\r\n          flex: 1;\r\n          overflow: hidden;\r\n        }\r\n\r\n        ::ng-deep .mat-mdc-tab-body {\r\n          height: 100%;\r\n        }\r\n\r\n        ::ng-deep .mat-mdc-tab-body-content {\r\n          height: 100%;\r\n          overflow: hidden;\r\n        }\r\n      }\r\n\r\n      .editor-layout {\r\n        display: flex;\r\n        height: 100%;\r\n        overflow: hidden;\r\n      }\r\n\r\n      .editor-sidebar {\r\n        height: 100%;\r\n        overflow-y: auto;\r\n        background: var(--ui-color-surface);\r\n        flex-shrink: 0;\r\n\r\n        &.left {\r\n          position: relative;\r\n          border-right: 1px solid var(--ui-color-border);\r\n          min-width: 200px;\r\n          max-width: 500px;\r\n        }\r\n\r\n        &.right {\r\n          position: relative;\r\n          border-left: 1px solid var(--ui-color-border);\r\n          min-width: 400px;\r\n          max-width: 1200px;\r\n        }\r\n      }\r\n\r\n      .editor-main {\r\n        flex: 1;\r\n        min-width: 0;\r\n        height: 100%;\r\n        overflow-y: auto;\r\n        padding: var(--ui-spacing-4);\r\n        background: var(--ui-color-bg-subtle);\r\n      }\r\n\r\n      .sections-container {\r\n        max-width: 1200px;\r\n        margin: 0 auto;\r\n        display: flex;\r\n        flex-direction: column;\r\n        gap: var(--ui-spacing-4);\r\n      }\r\n\r\n      .resize-handle {\r\n        position: absolute;\r\n        top: 0;\r\n        bottom: 0;\r\n        width: 6px;\r\n        cursor: ew-resize;\r\n        z-index: 10;\r\n        transition: background-color var(--ui-transition-fast);\r\n\r\n        /* Handle sinistro (per sidebar destra) */\r\n        left: 0;\r\n\r\n        &::before {\r\n          content: '';\r\n          position: absolute;\r\n          left: -3px;\r\n          top: 0;\r\n          bottom: 0;\r\n          width: 12px;\r\n        }\r\n\r\n        &:hover,\r\n        &.resizing {\r\n          background-color: color-mix(\r\n            in srgb,\r\n            var(--ui-color-primary) 30%,\r\n            transparent\r\n          );\r\n        }\r\n\r\n        &.resizing {\r\n          background-color: color-mix(\r\n            in srgb,\r\n            var(--ui-color-primary) 50%,\r\n            transparent\r\n          );\r\n        }\r\n      }\r\n\r\n      /* Handle destro (per sidebar sinistra) */\r\n      .resize-handle--right {\r\n        left: auto;\r\n        right: 0;\r\n\r\n        &::before {\r\n          left: auto;\r\n          right: -3px;\r\n        }\r\n      }\r\n\r\n      .empty-state {\r\n        display: flex;\r\n        flex-direction: column;\r\n        align-items: center;\r\n        justify-content: center;\r\n        padding: var(--ui-spacing-12) var(--ui-spacing-8);\r\n        text-align: center;\r\n        color: var(--ui-color-text-muted);\r\n\r\n        lucide-icon {\r\n          opacity: 0.4;\r\n        }\r\n\r\n        h3 {\r\n          margin: 0 0 var(--ui-spacing-2) 0;\r\n          font-size: var(--ui-font-size-xl);\r\n          font-weight: 400;\r\n          color: var(--ui-color-text-secondary);\r\n        }\r\n\r\n        p {\r\n          margin: 0 0 var(--ui-spacing-6) 0;\r\n          font-size: var(--ui-font-size-md);\r\n          color: var(--ui-color-text-muted);\r\n        }\r\n      }\r\n\r\n      /* Responsive */\r\n      @media (max-width: 1280px) {\r\n        .editor-sidebar.left {\r\n          width: 250px;\r\n        }\r\n      }\r\n\r\n      @media (max-width: 960px) {\r\n        .editor-layout {\r\n          flex-direction: column;\r\n        }\r\n\r\n        .editor-sidebar {\r\n          &.left {\r\n            display: none;\r\n          }\r\n\r\n          &.right {\r\n            position: fixed;\r\n            right: 0;\r\n            top: 64px;\r\n            width: 100% !important;\r\n            max-width: 350px;\r\n            height: calc(100vh - 64px);\r\n            z-index: 1000;\r\n            box-shadow: var(--ui-shadow-lg);\r\n            transform: translateX(100%);\r\n            transition: transform var(--ui-transition-normal);\r\n\r\n            .resize-handle {\r\n              display: none;\r\n            }\r\n          }\r\n        }\r\n      }\r\n    `,\r\n  ],\r\n})\r\nexport class UiFormBuilderEditorComponent implements OnInit, OnDestroy {\r\n  /** @internal Servizio per lo stato dell'editor. */\r\n  readonly stateService = inject(UiEditorStateService);\r\n\r\n  /** @internal Servizio per la persistenza. */\r\n  private readonly persistenceService = inject(UiEditorPersistenceService);\r\n\r\n  /** @internal Factory per i campi. */\r\n  private readonly fieldFactory = inject(UiEditorFieldFactoryService);\r\n\r\n  /** @internal Snack bar per i messaggi. */\r\n  private readonly snackBar = inject(MatSnackBar);\r\n\r\n  /** @internal Change detector. */\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n\r\n  /** @internal Subject per la distruzione. */\r\n  private readonly destroy$ = new Subject<void>();\r\n\r\n  /** Stato corrente dell'editor. */\r\n  state: UiEditorState;\r\n\r\n  /** Indice del tab selezionato (0=Editor, 1=Preview). */\r\n  selectedTabIndex = 0;\r\n\r\n  /** Breadcrumb di navigazione. */\r\n  breadcrumb: UiEditorBreadcrumbItem[] = [];\r\n\r\n  /** Se il componente e aperto in modalita dialog. */\r\n  isDialogMode = false;\r\n\r\n  /** Chiave per forzare il re-render del preview. */\r\n  previewKey = 0;\r\n\r\n  /** Campi disponibili per le condizioni (cache). */\r\n  availableFields: UiEditorAvailableField[] = [];\r\n\r\n  /** Larghezza della sidebar sinistra (px). */\r\n  leftSidebarWidth = 280;\r\n\r\n  /** Larghezza della sidebar destra (px). */\r\n  rightSidebarWidth = 460;\r\n\r\n  /** Se e in corso il resize della sidebar destra. */\r\n  isResizing = false;\r\n\r\n  /** Se e in corso il resize della sidebar sinistra. */\r\n  isResizingLeft = false;\r\n\r\n  /** @internal Limiti di resize sidebar destra. */\r\n  private readonly MIN_SIDEBAR_WIDTH = 400;\r\n  private readonly MAX_SIDEBAR_WIDTH = 1200;\r\n\r\n  /** @internal Limiti di resize sidebar sinistra. */\r\n  private readonly MIN_LEFT_SIDEBAR_WIDTH = 200;\r\n  private readonly MAX_LEFT_SIDEBAR_WIDTH = 500;\r\n\r\n  private resizeStartX = 0;\r\n  private resizeStartWidth = 0;\r\n  private readonly SIDEBAR_WIDTH_KEY = 'ui-editor-right-sidebar-width';\r\n  private readonly LEFT_SIDEBAR_WIDTH_KEY = 'ui-editor-left-sidebar-width';\r\n\r\n  constructor(\r\n    @Optional() public dialogRef: MatDialogRef<UiFormBuilderEditorComponent>,\r\n    @Optional() @Inject(MAT_DIALOG_DATA) public dialogData: UiEditorDialogData,\r\n  ) {\r\n    this.state = this.stateService.getCurrentState();\r\n    this.isDialogMode = !!this.dialogRef;\r\n\r\n    // Carica schema iniziale se fornito via dialog data\r\n    if (this.dialogData?.initialSchema) {\r\n      this.stateService.setSchema(this.dialogData.initialSchema);\r\n    }\r\n\r\n    this.loadSidebarWidth();\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    // Sottoscrizione allo stato\r\n    this.stateService.state$.pipe(takeUntil(this.destroy$)).subscribe((state) => {\r\n      this.state = state;\r\n      this.updateBreadcrumb();\r\n      this.updateAvailableFields();\r\n      this.cdr.markForCheck();\r\n    });\r\n\r\n    // Controlla salvataggi precedenti da recuperare\r\n    this.checkForRecovery();\r\n\r\n    // Auto-save ogni 30 secondi\r\n    interval(30000)\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe(() => {\r\n        if (this.state.isDirty) {\r\n          this.autoSave();\r\n        }\r\n      });\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  // ─── Toolbar actions ─────────────────────────────────────────\r\n\r\n  /** Aggiunge una nuova sezione. */\r\n  onAddSection(): void {\r\n    this.stateService.addSection();\r\n    this.showMessage('Sezione aggiunta');\r\n  }\r\n\r\n  /** Salva nel localStorage. */\r\n  onSave(): void {\r\n    try {\r\n      this.persistenceService.saveToLocalStorage(this.state.schema);\r\n      this.stateService.markAsSaved();\r\n      this.showMessage('Salvato con successo');\r\n    } catch (error: any) {\r\n      this.showMessage(error.message || 'Errore durante il salvataggio', true);\r\n    }\r\n  }\r\n\r\n  /** Crea un nuovo form. */\r\n  onNewForm(): void {\r\n    if (this.state.isDirty && !confirm('Ci sono modifiche non salvate. Continuare?')) {\r\n      return;\r\n    }\r\n    this.stateService.reset();\r\n    this.showMessage('Nuovo form creato');\r\n  }\r\n\r\n  /** Importa da file JSON. */\r\n  onImportJson(): void {\r\n    const input = document.createElement('input');\r\n    input.type = 'file';\r\n    input.accept = '.json';\r\n\r\n    input.onchange = async (e: any) => {\r\n      const file = e.target.files[0];\r\n      if (!file) return;\r\n\r\n      try {\r\n        const schema = await this.persistenceService.importFromJSON(file);\r\n        if (this.state.isDirty && !confirm('Ci sono modifiche non salvate. Sovrascrivere?')) {\r\n          return;\r\n        }\r\n        this.stateService.setSchema(schema);\r\n        this.showMessage('Schema importato con successo');\r\n      } catch (error: any) {\r\n        this.showMessage(error.message || \"Errore durante l'import\", true);\r\n      }\r\n    };\r\n\r\n    input.click();\r\n  }\r\n\r\n  /** Esporta come file JSON. */\r\n  onExportJson(): void {\r\n    try {\r\n      const validation = this.stateService.validateSchema();\r\n      if (!validation.valid) {\r\n        const message = `Schema con errori:\\n${validation.errors.map((e) => e.message).join('\\n')}`;\r\n        if (!confirm(`${message}\\n\\nEsportare comunque?`)) return;\r\n      }\r\n      this.persistenceService.exportToJSON(this.state.schema);\r\n      this.showMessage('Schema esportato con successo');\r\n    } catch (error: any) {\r\n      this.showMessage(error.message || \"Errore durante l'export\", true);\r\n    }\r\n  }\r\n\r\n  /** Pulisce tutto lo schema. */\r\n  onClear(): void {\r\n    if (!confirm('Vuoi davvero eliminare tutto? Questa azione non puo essere annullata.')) return;\r\n    this.persistenceService.createBackup(this.state.schema);\r\n    this.stateService.clearSchema();\r\n    this.showMessage('Schema pulito');\r\n  }\r\n\r\n  // ─── Sezioni ─────────────────────────────────────────────────\r\n\r\n  onSectionDuplicate(sectionId: string): void {\r\n    this.stateService.duplicateSection(sectionId);\r\n    this.showMessage('Sezione duplicata');\r\n  }\r\n\r\n  onSectionDelete(sectionId: string): void {\r\n    if (!confirm('Vuoi davvero eliminare questa sezione?')) return;\r\n    this.stateService.removeSection(sectionId);\r\n    this.showMessage('Sezione eliminata');\r\n  }\r\n\r\n  // ─── Campi ───────────────────────────────────────────────────\r\n\r\n  onFieldSelect(sectionId: string, fieldKey: string): void {\r\n    this.stateService.selectField(sectionId, fieldKey);\r\n  }\r\n\r\n  onFieldDuplicate(sectionId: string, fieldKey: string): void {\r\n    this.stateService.duplicateField(sectionId, fieldKey);\r\n    this.showMessage('Campo duplicato');\r\n  }\r\n\r\n  onFieldDelete(sectionId: string, fieldKey: string): void {\r\n    if (!confirm('Vuoi davvero eliminare questo campo?')) return;\r\n    this.stateService.removeField(sectionId, fieldKey);\r\n    this.showMessage('Campo eliminato');\r\n  }\r\n\r\n  onFieldUpdate(sectionId: string, fieldKey: string, updates: Partial<UiFormFieldDescriptor>): void {\r\n    this.stateService.updateField(sectionId, fieldKey, updates);\r\n  }\r\n\r\n  /** Gestisce il drop di un campo (dalla palette o tra sezioni). */\r\n  onFieldDrop(event: CdkDragDrop<any>): void {\r\n    const prevId = event.previousContainer.id;\r\n\r\n    if (prevId.startsWith('palette-')) {\r\n      // Drop dalla palette\r\n      const paletteItem: UiEditorFieldPaletteItem = event.item.data;\r\n      const section: UiFormSection = event.container.data;\r\n      const newField = this.fieldFactory.createDefaultField(paletteItem.type);\r\n      this.stateService.addFieldAtIndex(section.id, newField, event.currentIndex);\r\n      this.showMessage(`Campo ${paletteItem.label} aggiunto`);\r\n    } else if (event.previousContainer !== event.container) {\r\n      // Spostamento tra sezioni\r\n      const field: UiFormFieldDescriptor = event.item.data;\r\n      const fromSection: UiFormSection = event.previousContainer.data;\r\n      const toSection: UiFormSection = event.container.data;\r\n      this.stateService.moveFieldBetweenSections(fromSection.id, toSection.id, field.key, event.currentIndex);\r\n      this.showMessage('Campo spostato');\r\n    } else {\r\n      // Riordino nella stessa sezione\r\n      const section: UiFormSection = event.container.data;\r\n      this.stateService.moveFieldInSection(section.id, event.previousIndex, event.currentIndex);\r\n    }\r\n\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // ─── Computed properties ─────────────────────────────────────\r\n\r\n  /** Campo attualmente selezionato. */\r\n  get selectedField(): UiFormFieldDescriptor | null {\r\n    if (!this.state.selectedFieldKey) return null;\r\n    return this.stateService.findFieldByKey(this.state.selectedFieldKey);\r\n  }\r\n\r\n  /** ID della sezione del campo selezionato. */\r\n  get selectedFieldSectionId(): string | null {\r\n    if (!this.state.selectedFieldKey) return null;\r\n    return this.stateService.findSectionByFieldKey(this.state.selectedFieldKey)?.id || null;\r\n  }\r\n\r\n  // ─── Tab / Preview ───────────────────────────────────────────\r\n\r\n  onTabChange(index: number): void {\r\n    this.selectedTabIndex = index;\r\n    if (index === 1) this.refreshPreview();\r\n  }\r\n\r\n  refreshPreview(): void {\r\n    this.previewKey++;\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // ─── Dialog ──────────────────────────────────────────────────\r\n\r\n  /** Chiude la modale restituendo lo schema corrente. */\r\n  closeDialog(): void {\r\n    this.dialogRef?.close(this.state.schema);\r\n  }\r\n\r\n  // ─── Resize sidebar ──────────────────────────────────────────\r\n\r\n  onResizeStart(event: MouseEvent): void {\r\n    event.preventDefault();\r\n    this.isResizing = true;\r\n    this.resizeStartX = event.clientX;\r\n    this.resizeStartWidth = this.rightSidebarWidth;\r\n\r\n    document.addEventListener('mousemove', this.onResizeMove);\r\n    document.addEventListener('mouseup', this.onResizeEnd);\r\n    document.body.style.cursor = 'ew-resize';\r\n    document.body.style.userSelect = 'none';\r\n  }\r\n\r\n  private onResizeMove = (event: MouseEvent): void => {\r\n    if (!this.isResizing) return;\r\n    const delta = this.resizeStartX - event.clientX;\r\n    this.rightSidebarWidth = Math.max(\r\n      this.MIN_SIDEBAR_WIDTH,\r\n      Math.min(this.MAX_SIDEBAR_WIDTH, this.resizeStartWidth + delta),\r\n    );\r\n    this.cdr.markForCheck();\r\n  };\r\n\r\n  private onResizeEnd = (): void => {\r\n    if (!this.isResizing) return;\r\n    this.isResizing = false;\r\n    document.removeEventListener('mousemove', this.onResizeMove);\r\n    document.removeEventListener('mouseup', this.onResizeEnd);\r\n    document.body.style.cursor = '';\r\n    document.body.style.userSelect = '';\r\n    this.saveSidebarWidth();\r\n  };\r\n\r\n  // ─── Resize sidebar sinistra ─────────────────────────────────\r\n\r\n  onLeftResizeStart(event: MouseEvent): void {\r\n    event.preventDefault();\r\n    this.isResizingLeft = true;\r\n    this.resizeStartX = event.clientX;\r\n    this.resizeStartWidth = this.leftSidebarWidth;\r\n\r\n    document.addEventListener('mousemove', this.onLeftResizeMove);\r\n    document.addEventListener('mouseup', this.onLeftResizeEnd);\r\n    document.body.style.cursor = 'ew-resize';\r\n    document.body.style.userSelect = 'none';\r\n  }\r\n\r\n  private onLeftResizeMove = (event: MouseEvent): void => {\r\n    if (!this.isResizingLeft) return;\r\n    const delta = event.clientX - this.resizeStartX;\r\n    this.leftSidebarWidth = Math.max(\r\n      this.MIN_LEFT_SIDEBAR_WIDTH,\r\n      Math.min(this.MAX_LEFT_SIDEBAR_WIDTH, this.resizeStartWidth + delta),\r\n    );\r\n    this.cdr.markForCheck();\r\n  };\r\n\r\n  private onLeftResizeEnd = (): void => {\r\n    if (!this.isResizingLeft) return;\r\n    this.isResizingLeft = false;\r\n    document.removeEventListener('mousemove', this.onLeftResizeMove);\r\n    document.removeEventListener('mouseup', this.onLeftResizeEnd);\r\n    document.body.style.cursor = '';\r\n    document.body.style.userSelect = '';\r\n    this.saveLeftSidebarWidth();\r\n  };\r\n\r\n  // ─── Private ─────────────────────────────────────────────────\r\n\r\n  private checkForRecovery(): void {\r\n    if (this.persistenceService.hasLocalStorageData()) {\r\n      const age = this.persistenceService.getLocalStorageAge();\r\n      const message = age\r\n        ? `Trovato salvataggio del ${age.toLocaleString('it-IT')}. Vuoi recuperarlo?`\r\n        : 'Trovato salvataggio precedente. Vuoi recuperarlo?';\r\n\r\n      const ref = this.snackBar.open(message, 'Recupera', { duration: 10000 });\r\n      ref.onAction().subscribe(() => {\r\n        const schema = this.persistenceService.loadFromLocalStorage();\r\n        if (schema) {\r\n          this.stateService.setSchema(schema);\r\n          this.showMessage('Schema recuperato');\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  private autoSave(): void {\r\n    try {\r\n      this.persistenceService.saveToLocalStorage(this.state.schema);\r\n      this.stateService.markAsSaved();\r\n    } catch {\r\n      // Silenzioso per auto-save\r\n    }\r\n  }\r\n\r\n  private updateBreadcrumb(): void {\r\n    this.breadcrumb = [\r\n      {\r\n        label: this.state.schema.title || 'Form',\r\n        id: this.state.schema.id,\r\n        type: 'form',\r\n        active: !this.state.selectedSectionId && !this.state.selectedFieldKey,\r\n      },\r\n    ];\r\n\r\n    if (this.state.selectedSectionId) {\r\n      const section = this.state.schema.sections.find((s) => s.id === this.state.selectedSectionId);\r\n      if (section) {\r\n        this.breadcrumb.push({\r\n          label: section.title || 'Sezione',\r\n          id: section.id,\r\n          type: 'section',\r\n          active: !this.state.selectedFieldKey,\r\n        });\r\n      }\r\n    }\r\n\r\n    if (this.state.selectedFieldKey) {\r\n      const field = this.stateService.findFieldByKey(this.state.selectedFieldKey);\r\n      if (field) {\r\n        this.breadcrumb.push({\r\n          label: field.label,\r\n          id: field.key,\r\n          type: 'field',\r\n          active: true,\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  private updateAvailableFields(): void {\r\n    const fields: UiEditorAvailableField[] = [];\r\n    this.state.schema.sections.forEach((section) => {\r\n      section.fields.forEach((field) => {\r\n        const af: UiEditorAvailableField = {\r\n          key: field.key,\r\n          label: field.label,\r\n          type: field.type,\r\n        };\r\n        if (field.options && Array.isArray(field.options) && field.options.length > 0) {\r\n          af.options = field.options.map((opt) => ({\r\n            value: opt.value,\r\n            label: opt.label,\r\n          }));\r\n        }\r\n        fields.push(af);\r\n      });\r\n    });\r\n    this.availableFields = fields;\r\n  }\r\n\r\n  private showMessage(message: string, isError = false): void {\r\n    this.snackBar.open(message, 'Chiudi', {\r\n      duration: 3000,\r\n      panelClass: isError ? ['ui-snackbar-error'] : [],\r\n    });\r\n  }\r\n\r\n  private loadSidebarWidth(): void {\r\n    try {\r\n      const savedRight = localStorage.getItem(this.SIDEBAR_WIDTH_KEY);\r\n      if (savedRight) {\r\n        const w = parseInt(savedRight, 10);\r\n        if (!isNaN(w) && w >= this.MIN_SIDEBAR_WIDTH && w <= this.MAX_SIDEBAR_WIDTH) {\r\n          this.rightSidebarWidth = w;\r\n        }\r\n      }\r\n      const savedLeft = localStorage.getItem(this.LEFT_SIDEBAR_WIDTH_KEY);\r\n      if (savedLeft) {\r\n        const w = parseInt(savedLeft, 10);\r\n        if (!isNaN(w) && w >= this.MIN_LEFT_SIDEBAR_WIDTH && w <= this.MAX_LEFT_SIDEBAR_WIDTH) {\r\n          this.leftSidebarWidth = w;\r\n        }\r\n      }\r\n    } catch {\r\n      // Silenzioso\r\n    }\r\n  }\r\n\r\n  private saveSidebarWidth(): void {\r\n    try {\r\n      localStorage.setItem(this.SIDEBAR_WIDTH_KEY, this.rightSidebarWidth.toString());\r\n    } catch {\r\n      // Silenzioso\r\n    }\r\n  }\r\n\r\n  private saveLeftSidebarWidth(): void {\r\n    try {\r\n      localStorage.setItem(this.LEFT_SIDEBAR_WIDTH_KEY, this.leftSidebarWidth.toString());\r\n    } catch {\r\n      // Silenzioso\r\n    }\r\n  }\r\n}\r\n"]}
|