@gnggln/ng-ui-system 1.0.0-alpha.0

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