@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,472 @@
1
+ /**
2
+ * @module ng-ui-system/form-builder-editor
3
+ * Componente editor di sezione per il form builder.
4
+ *
5
+ * Renderizza una singola sezione del form come card, con supporto
6
+ * per espansione/collasso, editing inline del titolo, drag & drop
7
+ * dei campi e azioni CRUD su sezione e campi.
8
+ */
9
+ import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter } from '@angular/core';
10
+ import { FormsModule } from '@angular/forms';
11
+ import { MatCardModule } from '@angular/material/card';
12
+ import { MatFormFieldModule } from '@angular/material/form-field';
13
+ import { MatInputModule } from '@angular/material/input';
14
+ import { DragDropModule } from '@angular/cdk/drag-drop';
15
+ import { LucideAngularModule } from 'lucide-angular';
16
+ import { UiButtonComponent } from '../../../button/index';
17
+ import * as i0 from "@angular/core";
18
+ import * as i1 from "@angular/forms";
19
+ import * as i2 from "@angular/material/card";
20
+ import * as i3 from "@angular/material/form-field";
21
+ import * as i4 from "@angular/material/input";
22
+ import * as i5 from "@angular/cdk/drag-drop";
23
+ import * as i6 from "lucide-angular";
24
+ /**
25
+ * Mappa tipo di campo -> nome icona Lucide.
26
+ * Usata per visualizzare l'icona corretta accanto a ciascun campo.
27
+ */
28
+ const FIELD_ICON_MAP = {
29
+ text: 'type',
30
+ email: 'mail',
31
+ password: 'lock', // NOSONAR - Icon name for password field type, not a credential
32
+ number: 'hash',
33
+ textarea: 'text',
34
+ select: 'chevron-down',
35
+ multiselect: 'list',
36
+ freemultiselect: 'list-plus',
37
+ radio: 'circle-dot',
38
+ checkbox: 'check-square',
39
+ switch: 'toggle-left',
40
+ date: 'calendar',
41
+ datetime: 'clock',
42
+ file: 'upload',
43
+ flag: 'flag',
44
+ custom: 'settings',
45
+ divider: 'minus',
46
+ location: 'map-pin',
47
+ 'location-table': 'map',
48
+ };
49
+ /**
50
+ * Editor visuale di una singola sezione del form.
51
+ *
52
+ * Supporta:
53
+ * - Header espandibile/collassabile con editing inline del titolo
54
+ * - Pulsanti azione: modifica titolo, duplica, elimina
55
+ * - Drop list CDK per ricevere campi dalla palette o da altre sezioni
56
+ * - Ogni campo mostra: handle di trascinamento, icona tipo, label, chiave, badge tipo, asterisco required
57
+ * - Azioni campo (al passaggio mouse): duplica, elimina
58
+ * - Stato vuoto quando non ci sono campi
59
+ */
60
+ export class UiSectionEditorComponent {
61
+ constructor() {
62
+ /** Indica se la sezione e attualmente selezionata. */
63
+ this.isSelected = false;
64
+ /** Chiave del campo attualmente selezionato (o null). */
65
+ this.selectedFieldKey = null;
66
+ /** Emette aggiornamenti parziali alla sezione. */
67
+ this.sectionUpdate = new EventEmitter();
68
+ /** Emette richiesta di eliminazione della sezione. */
69
+ this.sectionDelete = new EventEmitter();
70
+ /** Emette richiesta di duplicazione della sezione. */
71
+ this.sectionDuplicate = new EventEmitter();
72
+ /** Emette la chiave del campo selezionato. */
73
+ this.fieldSelect = new EventEmitter();
74
+ /** Emette la chiave del campo da duplicare. */
75
+ this.fieldDuplicate = new EventEmitter();
76
+ /** Emette la chiave del campo da eliminare. */
77
+ this.fieldDelete = new EventEmitter();
78
+ /** Emette l'evento di drop del CDK drag & drop. */
79
+ this.fieldDrop = new EventEmitter();
80
+ /** Stato di espansione della sezione. */
81
+ this.isExpanded = true;
82
+ /** Indica se il titolo e in fase di modifica inline. */
83
+ this.isEditingTitle = false;
84
+ /** Titolo temporaneo durante la modifica inline. */
85
+ this.tempTitle = '';
86
+ }
87
+ /**
88
+ * Alterna lo stato di espansione/collasso della sezione.
89
+ */
90
+ toggleExpand() {
91
+ this.isExpanded = !this.isExpanded;
92
+ }
93
+ /**
94
+ * Avvia la modifica inline del titolo della sezione.
95
+ * @param event - Evento del mouse per prevenire la propagazione.
96
+ */
97
+ startTitleEdit(event) {
98
+ event.stopPropagation();
99
+ this.tempTitle = this.section.title || '';
100
+ this.isEditingTitle = true;
101
+ }
102
+ /**
103
+ * Conferma la modifica del titolo ed emette l'aggiornamento.
104
+ */
105
+ confirmTitleEdit() {
106
+ if (this.isEditingTitle) {
107
+ this.isEditingTitle = false;
108
+ const trimmed = this.tempTitle.trim();
109
+ if (trimmed && trimmed !== this.section.title) {
110
+ this.sectionUpdate.emit({ title: trimmed });
111
+ }
112
+ }
113
+ }
114
+ /**
115
+ * Annulla la modifica del titolo, ripristinando il valore originale.
116
+ */
117
+ cancelTitleEdit() {
118
+ this.isEditingTitle = false;
119
+ this.tempTitle = '';
120
+ }
121
+ /**
122
+ * Restituisce il nome dell'icona Lucide associata al tipo di campo.
123
+ * @param type - Tipo di campo del form.
124
+ * @returns Nome dell'icona Lucide.
125
+ */
126
+ getFieldIcon(type) {
127
+ return FIELD_ICON_MAP[type] || 'help-circle';
128
+ }
129
+ /**
130
+ * Gestisce l'evento di drop del CDK drag & drop e lo propaga al parent.
131
+ * @param event - Evento CdkDragDrop.
132
+ */
133
+ onFieldDrop(event) {
134
+ this.fieldDrop.emit(event);
135
+ }
136
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiSectionEditorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
137
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiSectionEditorComponent, isStandalone: true, selector: "ui-section-editor", inputs: { section: "section", isSelected: "isSelected", selectedFieldKey: "selectedFieldKey" }, outputs: { sectionUpdate: "sectionUpdate", sectionDelete: "sectionDelete", sectionDuplicate: "sectionDuplicate", fieldSelect: "fieldSelect", fieldDuplicate: "fieldDuplicate", fieldDelete: "fieldDelete", fieldDrop: "fieldDrop" }, ngImport: i0, template: `
138
+ <!-- Card della sezione -->
139
+ <mat-card
140
+ class="section-card"
141
+ [class.section-card--selected]="isSelected"
142
+ >
143
+ <!-- Header della sezione -->
144
+ <div class="section-header" (click)="toggleExpand()">
145
+ <!-- Indicatore espansione -->
146
+ <ui-button
147
+ [icon]="isExpanded ? 'chevron-down' : 'chevron-right'"
148
+ variant="ghost"
149
+ size="sm"
150
+ [tooltip]="isExpanded ? 'Comprimi sezione' : 'Espandi sezione'"
151
+ ariaLabel="Espandi/Comprimi sezione"
152
+ />
153
+
154
+ <!-- Titolo della sezione (doppio click per modificare) -->
155
+ @if (isEditingTitle) {
156
+ <mat-form-field class="section-header__title-field" appearance="outline">
157
+ <input
158
+ matInput
159
+ [(ngModel)]="tempTitle"
160
+ (blur)="confirmTitleEdit()"
161
+ (keydown.enter)="confirmTitleEdit()"
162
+ (keydown.escape)="cancelTitleEdit()"
163
+ (click)="$event.stopPropagation()"
164
+ autofocus
165
+ />
166
+ </mat-form-field>
167
+ } @else {
168
+ <span
169
+ class="section-header__title"
170
+ (dblclick)="startTitleEdit($event)"
171
+ >
172
+ {{ section.title || 'Sezione senza titolo' }}
173
+ </span>
174
+ }
175
+
176
+ <!-- Badge conteggio campi -->
177
+ <span class="section-header__badge">
178
+ {{ section.fields.length }} {{ section.fields.length === 1 ? 'campo' : 'campi' }}
179
+ </span>
180
+
181
+ <!-- Azioni sezione -->
182
+ <div class="section-header__actions" (click)="$event.stopPropagation()">
183
+ <ui-button
184
+ icon="pencil"
185
+ variant="ghost"
186
+ size="sm"
187
+ ariaLabel="Modifica titolo"
188
+ tooltip="Modifica titolo"
189
+ (click)="startTitleEdit($event)"
190
+ />
191
+
192
+ <ui-button
193
+ icon="copy"
194
+ variant="ghost"
195
+ size="sm"
196
+ ariaLabel="Duplica sezione"
197
+ tooltip="Duplica sezione"
198
+ (click)="sectionDuplicate.emit()"
199
+ />
200
+
201
+ <ui-button
202
+ icon="trash-2"
203
+ variant="warn"
204
+ size="sm"
205
+ ariaLabel="Elimina sezione"
206
+ tooltip="Elimina sezione"
207
+ (click)="sectionDelete.emit()"
208
+ />
209
+ </div>
210
+ </div>
211
+
212
+ <!-- Corpo della sezione (espandibile) -->
213
+ @if (isExpanded) {
214
+ <div class="section-body">
215
+ <!-- Drop list CDK per i campi -->
216
+ <div
217
+ cdkDropList
218
+ [cdkDropListData]="section"
219
+ [id]="'section-' + section.id"
220
+ (cdkDropListDropped)="onFieldDrop($event)"
221
+ class="field-drop-list"
222
+ >
223
+ @for (field of section.fields; track field.key) {
224
+ <div
225
+ class="field-item"
226
+ cdkDrag
227
+ [cdkDragData]="field"
228
+ [class.field-item--selected]="field.key === selectedFieldKey"
229
+ (click)="fieldSelect.emit(field.key)"
230
+ >
231
+ <!-- Handle di trascinamento -->
232
+ <div class="field-item__handle" cdkDragHandle>
233
+ <lucide-icon name="grip-vertical" [size]="16" />
234
+ </div>
235
+
236
+ <!-- Icona tipo campo -->
237
+ <div class="field-item__icon">
238
+ <lucide-icon [name]="getFieldIcon(field.type)" [size]="16" />
239
+ </div>
240
+
241
+ <!-- Informazioni campo -->
242
+ <div class="field-item__info">
243
+ <span class="field-item__label">
244
+ {{ field.label || 'Senza etichetta' }}
245
+ @if (field.required) {
246
+ <span class="field-item__required">*</span>
247
+ }
248
+ </span>
249
+ <span class="field-item__key">{{ field.key }}</span>
250
+ </div>
251
+
252
+ <!-- Badge tipo campo -->
253
+ <span class="field-item__type-badge">{{ field.type }}</span>
254
+
255
+ <!-- Azioni campo (visibili al passaggio del mouse) -->
256
+ <div class="field-item__actions" (click)="$event.stopPropagation()">
257
+ <ui-button
258
+ icon="copy"
259
+ variant="ghost"
260
+ size="xs"
261
+ ariaLabel="Duplica campo"
262
+ tooltip="Duplica campo"
263
+ (click)="fieldDuplicate.emit(field.key)"
264
+ />
265
+
266
+ <ui-button
267
+ icon="trash-2"
268
+ variant="warn"
269
+ size="xs"
270
+ ariaLabel="Elimina campo"
271
+ tooltip="Elimina campo"
272
+ (click)="fieldDelete.emit(field.key)"
273
+ />
274
+ </div>
275
+ </div>
276
+ } @empty {
277
+ <!-- Stato vuoto -->
278
+ <div class="field-drop-list__empty">
279
+ <lucide-icon name="mouse-pointer-click" [size]="24" />
280
+ <span>Trascina qui i campi dalla palette</span>
281
+ </div>
282
+ }
283
+ </div>
284
+ </div>
285
+ }
286
+ </mat-card>
287
+ `, isInline: true, styles: [":host{display:block}.section-card{border:2px solid var(--ui-color-border);border-radius:var(--ui-radius-md);overflow:hidden;transition:border-color var(--ui-transition-fast),box-shadow var(--ui-transition-fast)}.section-card:hover{box-shadow:var(--ui-shadow-sm)}.section-card--selected{border-color:var(--ui-color-primary);box-shadow:var(--ui-shadow-md)}.section-header{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-bg-subtle);cursor:pointer;-webkit-user-select:none;user-select:none;transition:background var(--ui-transition-fast)}.section-header:hover{background:var(--ui-color-surface)}.section-header__title{flex:1;font-size:var(--ui-font-size-md);font-weight:600;color:var(--ui-color-text);cursor:pointer}.section-header__title-field{flex:1}.section-header__title-field ::ng-deep .mat-mdc-form-field-infix{padding-top:var(--ui-spacing-1)!important;padding-bottom:var(--ui-spacing-1)!important;min-height:unset}.section-header__badge{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);background:var(--ui-color-surface);padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);white-space:nowrap}.section-header__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);opacity:0;transition:opacity var(--ui-transition-fast)}.section-header:hover .section-header__actions{opacity:1}.section-body{padding:var(--ui-spacing-3)}.field-drop-list{min-height:48px;border:2px dashed var(--ui-color-border);border-radius:var(--ui-radius-sm);transition:border-color var(--ui-transition-fast),background var(--ui-transition-fast)}.field-drop-list.cdk-drop-list-dragging{border-color:var(--ui-color-primary);background:var(--ui-color-primary-bg)}.field-drop-list__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-6) var(--ui-spacing-4);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm)}.field-item{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);cursor:pointer;transition:background var(--ui-transition-fast)}.field-item:first-child{border-top-left-radius:var(--ui-radius-sm);border-top-right-radius:var(--ui-radius-sm)}.field-item:last-child{border-bottom:none;border-bottom-left-radius:var(--ui-radius-sm);border-bottom-right-radius:var(--ui-radius-sm)}.field-item:hover{background:var(--ui-color-bg-subtle)}.field-item--selected{background:var(--ui-color-primary-bg);border-left:3px solid var(--ui-color-primary)}.field-item__handle{display:flex;align-items:center;cursor:grab;color:var(--ui-color-text-muted);transition:color var(--ui-transition-fast)}.field-item__handle:active{cursor:grabbing}.field-item:hover .field-item__handle{color:var(--ui-color-text-secondary)}.field-item__icon{display:flex;align-items:center;justify-content:center;width:28px;height:28px;border-radius:var(--ui-radius-sm);background:var(--ui-color-bg-subtle);color:var(--ui-color-primary)}.field-item__info{flex:1;display:flex;flex-direction:column;min-width:0}.field-item__label{font-size:var(--ui-font-size-sm);font-weight:500;color:var(--ui-color-text);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-item__required{color:var(--ui-color-warn);margin-left:var(--ui-spacing-1)}.field-item__key{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:monospace;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-item__type-badge{font-size:var(--ui-font-size-xs);color:var(--ui-color-primary);background:var(--ui-color-primary-bg);padding:2px var(--ui-spacing-2);border-radius:var(--ui-radius-sm);white-space:nowrap;font-weight:500}.field-item__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);opacity:0;transition:opacity var(--ui-transition-fast)}.field-item:hover .field-item__actions{opacity:1}.cdk-drag-preview{box-sizing:border-box;border-radius:var(--ui-radius-sm);box-shadow:var(--ui-shadow-md);background:var(--ui-color-surface);padding:var(--ui-spacing-2) var(--ui-spacing-3);display:flex;align-items:center;gap:var(--ui-spacing-2)}.cdk-drag-placeholder{opacity:.3}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatCardModule }, { kind: "component", type: i2.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: DragDropModule }, { kind: "directive", type: i5.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep", "cdkDropListElementContainer"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i5.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer", "cdkDragScale"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "directive", type: i5.CdkDragHandle, selector: "[cdkDragHandle]", inputs: ["cdkDragHandleDisabled"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i6.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
288
+ }
289
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiSectionEditorComponent, decorators: [{
290
+ type: Component,
291
+ args: [{ selector: 'ui-section-editor', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [
292
+ FormsModule,
293
+ MatCardModule,
294
+ MatFormFieldModule,
295
+ MatInputModule,
296
+ DragDropModule,
297
+ LucideAngularModule,
298
+ UiButtonComponent,
299
+ ], template: `
300
+ <!-- Card della sezione -->
301
+ <mat-card
302
+ class="section-card"
303
+ [class.section-card--selected]="isSelected"
304
+ >
305
+ <!-- Header della sezione -->
306
+ <div class="section-header" (click)="toggleExpand()">
307
+ <!-- Indicatore espansione -->
308
+ <ui-button
309
+ [icon]="isExpanded ? 'chevron-down' : 'chevron-right'"
310
+ variant="ghost"
311
+ size="sm"
312
+ [tooltip]="isExpanded ? 'Comprimi sezione' : 'Espandi sezione'"
313
+ ariaLabel="Espandi/Comprimi sezione"
314
+ />
315
+
316
+ <!-- Titolo della sezione (doppio click per modificare) -->
317
+ @if (isEditingTitle) {
318
+ <mat-form-field class="section-header__title-field" appearance="outline">
319
+ <input
320
+ matInput
321
+ [(ngModel)]="tempTitle"
322
+ (blur)="confirmTitleEdit()"
323
+ (keydown.enter)="confirmTitleEdit()"
324
+ (keydown.escape)="cancelTitleEdit()"
325
+ (click)="$event.stopPropagation()"
326
+ autofocus
327
+ />
328
+ </mat-form-field>
329
+ } @else {
330
+ <span
331
+ class="section-header__title"
332
+ (dblclick)="startTitleEdit($event)"
333
+ >
334
+ {{ section.title || 'Sezione senza titolo' }}
335
+ </span>
336
+ }
337
+
338
+ <!-- Badge conteggio campi -->
339
+ <span class="section-header__badge">
340
+ {{ section.fields.length }} {{ section.fields.length === 1 ? 'campo' : 'campi' }}
341
+ </span>
342
+
343
+ <!-- Azioni sezione -->
344
+ <div class="section-header__actions" (click)="$event.stopPropagation()">
345
+ <ui-button
346
+ icon="pencil"
347
+ variant="ghost"
348
+ size="sm"
349
+ ariaLabel="Modifica titolo"
350
+ tooltip="Modifica titolo"
351
+ (click)="startTitleEdit($event)"
352
+ />
353
+
354
+ <ui-button
355
+ icon="copy"
356
+ variant="ghost"
357
+ size="sm"
358
+ ariaLabel="Duplica sezione"
359
+ tooltip="Duplica sezione"
360
+ (click)="sectionDuplicate.emit()"
361
+ />
362
+
363
+ <ui-button
364
+ icon="trash-2"
365
+ variant="warn"
366
+ size="sm"
367
+ ariaLabel="Elimina sezione"
368
+ tooltip="Elimina sezione"
369
+ (click)="sectionDelete.emit()"
370
+ />
371
+ </div>
372
+ </div>
373
+
374
+ <!-- Corpo della sezione (espandibile) -->
375
+ @if (isExpanded) {
376
+ <div class="section-body">
377
+ <!-- Drop list CDK per i campi -->
378
+ <div
379
+ cdkDropList
380
+ [cdkDropListData]="section"
381
+ [id]="'section-' + section.id"
382
+ (cdkDropListDropped)="onFieldDrop($event)"
383
+ class="field-drop-list"
384
+ >
385
+ @for (field of section.fields; track field.key) {
386
+ <div
387
+ class="field-item"
388
+ cdkDrag
389
+ [cdkDragData]="field"
390
+ [class.field-item--selected]="field.key === selectedFieldKey"
391
+ (click)="fieldSelect.emit(field.key)"
392
+ >
393
+ <!-- Handle di trascinamento -->
394
+ <div class="field-item__handle" cdkDragHandle>
395
+ <lucide-icon name="grip-vertical" [size]="16" />
396
+ </div>
397
+
398
+ <!-- Icona tipo campo -->
399
+ <div class="field-item__icon">
400
+ <lucide-icon [name]="getFieldIcon(field.type)" [size]="16" />
401
+ </div>
402
+
403
+ <!-- Informazioni campo -->
404
+ <div class="field-item__info">
405
+ <span class="field-item__label">
406
+ {{ field.label || 'Senza etichetta' }}
407
+ @if (field.required) {
408
+ <span class="field-item__required">*</span>
409
+ }
410
+ </span>
411
+ <span class="field-item__key">{{ field.key }}</span>
412
+ </div>
413
+
414
+ <!-- Badge tipo campo -->
415
+ <span class="field-item__type-badge">{{ field.type }}</span>
416
+
417
+ <!-- Azioni campo (visibili al passaggio del mouse) -->
418
+ <div class="field-item__actions" (click)="$event.stopPropagation()">
419
+ <ui-button
420
+ icon="copy"
421
+ variant="ghost"
422
+ size="xs"
423
+ ariaLabel="Duplica campo"
424
+ tooltip="Duplica campo"
425
+ (click)="fieldDuplicate.emit(field.key)"
426
+ />
427
+
428
+ <ui-button
429
+ icon="trash-2"
430
+ variant="warn"
431
+ size="xs"
432
+ ariaLabel="Elimina campo"
433
+ tooltip="Elimina campo"
434
+ (click)="fieldDelete.emit(field.key)"
435
+ />
436
+ </div>
437
+ </div>
438
+ } @empty {
439
+ <!-- Stato vuoto -->
440
+ <div class="field-drop-list__empty">
441
+ <lucide-icon name="mouse-pointer-click" [size]="24" />
442
+ <span>Trascina qui i campi dalla palette</span>
443
+ </div>
444
+ }
445
+ </div>
446
+ </div>
447
+ }
448
+ </mat-card>
449
+ `, styles: [":host{display:block}.section-card{border:2px solid var(--ui-color-border);border-radius:var(--ui-radius-md);overflow:hidden;transition:border-color var(--ui-transition-fast),box-shadow var(--ui-transition-fast)}.section-card:hover{box-shadow:var(--ui-shadow-sm)}.section-card--selected{border-color:var(--ui-color-primary);box-shadow:var(--ui-shadow-md)}.section-header{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-bg-subtle);cursor:pointer;-webkit-user-select:none;user-select:none;transition:background var(--ui-transition-fast)}.section-header:hover{background:var(--ui-color-surface)}.section-header__title{flex:1;font-size:var(--ui-font-size-md);font-weight:600;color:var(--ui-color-text);cursor:pointer}.section-header__title-field{flex:1}.section-header__title-field ::ng-deep .mat-mdc-form-field-infix{padding-top:var(--ui-spacing-1)!important;padding-bottom:var(--ui-spacing-1)!important;min-height:unset}.section-header__badge{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);background:var(--ui-color-surface);padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);white-space:nowrap}.section-header__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);opacity:0;transition:opacity var(--ui-transition-fast)}.section-header:hover .section-header__actions{opacity:1}.section-body{padding:var(--ui-spacing-3)}.field-drop-list{min-height:48px;border:2px dashed var(--ui-color-border);border-radius:var(--ui-radius-sm);transition:border-color var(--ui-transition-fast),background var(--ui-transition-fast)}.field-drop-list.cdk-drop-list-dragging{border-color:var(--ui-color-primary);background:var(--ui-color-primary-bg)}.field-drop-list__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-6) var(--ui-spacing-4);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm)}.field-item{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);cursor:pointer;transition:background var(--ui-transition-fast)}.field-item:first-child{border-top-left-radius:var(--ui-radius-sm);border-top-right-radius:var(--ui-radius-sm)}.field-item:last-child{border-bottom:none;border-bottom-left-radius:var(--ui-radius-sm);border-bottom-right-radius:var(--ui-radius-sm)}.field-item:hover{background:var(--ui-color-bg-subtle)}.field-item--selected{background:var(--ui-color-primary-bg);border-left:3px solid var(--ui-color-primary)}.field-item__handle{display:flex;align-items:center;cursor:grab;color:var(--ui-color-text-muted);transition:color var(--ui-transition-fast)}.field-item__handle:active{cursor:grabbing}.field-item:hover .field-item__handle{color:var(--ui-color-text-secondary)}.field-item__icon{display:flex;align-items:center;justify-content:center;width:28px;height:28px;border-radius:var(--ui-radius-sm);background:var(--ui-color-bg-subtle);color:var(--ui-color-primary)}.field-item__info{flex:1;display:flex;flex-direction:column;min-width:0}.field-item__label{font-size:var(--ui-font-size-sm);font-weight:500;color:var(--ui-color-text);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-item__required{color:var(--ui-color-warn);margin-left:var(--ui-spacing-1)}.field-item__key{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:monospace;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-item__type-badge{font-size:var(--ui-font-size-xs);color:var(--ui-color-primary);background:var(--ui-color-primary-bg);padding:2px var(--ui-spacing-2);border-radius:var(--ui-radius-sm);white-space:nowrap;font-weight:500}.field-item__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);opacity:0;transition:opacity var(--ui-transition-fast)}.field-item:hover .field-item__actions{opacity:1}.cdk-drag-preview{box-sizing:border-box;border-radius:var(--ui-radius-sm);box-shadow:var(--ui-shadow-md);background:var(--ui-color-surface);padding:var(--ui-spacing-2) var(--ui-spacing-3);display:flex;align-items:center;gap:var(--ui-spacing-2)}.cdk-drag-placeholder{opacity:.3}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"] }]
450
+ }], propDecorators: { section: [{
451
+ type: Input,
452
+ args: [{ required: true }]
453
+ }], isSelected: [{
454
+ type: Input
455
+ }], selectedFieldKey: [{
456
+ type: Input
457
+ }], sectionUpdate: [{
458
+ type: Output
459
+ }], sectionDelete: [{
460
+ type: Output
461
+ }], sectionDuplicate: [{
462
+ type: Output
463
+ }], fieldSelect: [{
464
+ type: Output
465
+ }], fieldDuplicate: [{
466
+ type: Output
467
+ }], fieldDelete: [{
468
+ type: Output
469
+ }], fieldDrop: [{
470
+ type: Output
471
+ }] } });
472
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"section-editor.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/form-builder-editor\r\n * Componente editor di sezione per il form builder.\r\n *\r\n * Renderizza una singola sezione del form come card, con supporto\r\n * per espansione/collasso, editing inline del titolo, drag & drop\r\n * dei campi e azioni CRUD su sezione e campi.\r\n */\r\n\r\nimport { Component, ChangeDetectionStrategy, Input, Output, EventEmitter } from '@angular/core';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { MatCardModule } from '@angular/material/card';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { CdkDragDrop, DragDropModule } from '@angular/cdk/drag-drop';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\n\r\nimport { UiButtonComponent } from '../../../button/index';\r\nimport { UiFormSection } from '../../../form-builder/types/index';\r\nimport { UiFieldType } from '../../../form-builder/types/index';\r\n\r\n/**\r\n * Mappa tipo di campo -> nome icona Lucide.\r\n * Usata per visualizzare l'icona corretta accanto a ciascun campo.\r\n */\r\nconst FIELD_ICON_MAP: Record<UiFieldType, string> = {\r\n  text: 'type',\r\n  email: 'mail',\r\n  password: 'lock', // NOSONAR - Icon name for password field type, not a credential\r\n  number: 'hash',\r\n  textarea: 'text',\r\n  select: 'chevron-down',\r\n  multiselect: 'list',\r\n  freemultiselect: 'list-plus',\r\n  radio: 'circle-dot',\r\n  checkbox: 'check-square',\r\n  switch: 'toggle-left',\r\n  date: 'calendar',\r\n  datetime: 'clock',\r\n  file: 'upload',\r\n  flag: 'flag',\r\n  custom: 'settings',\r\n  divider: 'minus',\r\n  location: 'map-pin',\r\n  'location-table': 'map',\r\n};\r\n\r\n/**\r\n * Editor visuale di una singola sezione del form.\r\n *\r\n * Supporta:\r\n * - Header espandibile/collassabile con editing inline del titolo\r\n * - Pulsanti azione: modifica titolo, duplica, elimina\r\n * - Drop list CDK per ricevere campi dalla palette o da altre sezioni\r\n * - Ogni campo mostra: handle di trascinamento, icona tipo, label, chiave, badge tipo, asterisco required\r\n * - Azioni campo (al passaggio mouse): duplica, elimina\r\n * - Stato vuoto quando non ci sono campi\r\n */\r\n@Component({\r\n  selector: 'ui-section-editor',\r\n  standalone: true,\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  imports: [\r\n    FormsModule,\r\n    MatCardModule,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    DragDropModule,\r\n    LucideAngularModule,\r\n    UiButtonComponent,\r\n  ],\r\n  template: `\r\n    <!-- Card della sezione -->\r\n    <mat-card\r\n      class=\"section-card\"\r\n      [class.section-card--selected]=\"isSelected\"\r\n    >\r\n      <!-- Header della sezione -->\r\n      <div class=\"section-header\" (click)=\"toggleExpand()\">\r\n        <!-- Indicatore espansione -->\r\n        <ui-button\r\n          [icon]=\"isExpanded ? 'chevron-down' : 'chevron-right'\"\r\n          variant=\"ghost\"\r\n          size=\"sm\"\r\n          [tooltip]=\"isExpanded ? 'Comprimi sezione' : 'Espandi sezione'\"\r\n          ariaLabel=\"Espandi/Comprimi sezione\"\r\n        />\r\n\r\n        <!-- Titolo della sezione (doppio click per modificare) -->\r\n        @if (isEditingTitle) {\r\n          <mat-form-field class=\"section-header__title-field\" appearance=\"outline\">\r\n            <input\r\n              matInput\r\n              [(ngModel)]=\"tempTitle\"\r\n              (blur)=\"confirmTitleEdit()\"\r\n              (keydown.enter)=\"confirmTitleEdit()\"\r\n              (keydown.escape)=\"cancelTitleEdit()\"\r\n              (click)=\"$event.stopPropagation()\"\r\n              autofocus\r\n            />\r\n          </mat-form-field>\r\n        } @else {\r\n          <span\r\n            class=\"section-header__title\"\r\n            (dblclick)=\"startTitleEdit($event)\"\r\n          >\r\n            {{ section.title || 'Sezione senza titolo' }}\r\n          </span>\r\n        }\r\n\r\n        <!-- Badge conteggio campi -->\r\n        <span class=\"section-header__badge\">\r\n          {{ section.fields.length }} {{ section.fields.length === 1 ? 'campo' : 'campi' }}\r\n        </span>\r\n\r\n        <!-- Azioni sezione -->\r\n        <div class=\"section-header__actions\" (click)=\"$event.stopPropagation()\">\r\n          <ui-button\r\n            icon=\"pencil\"\r\n            variant=\"ghost\"\r\n            size=\"sm\"\r\n            ariaLabel=\"Modifica titolo\"\r\n            tooltip=\"Modifica titolo\"\r\n            (click)=\"startTitleEdit($event)\"\r\n          />\r\n\r\n          <ui-button\r\n            icon=\"copy\"\r\n            variant=\"ghost\"\r\n            size=\"sm\"\r\n            ariaLabel=\"Duplica sezione\"\r\n            tooltip=\"Duplica sezione\"\r\n            (click)=\"sectionDuplicate.emit()\"\r\n          />\r\n\r\n          <ui-button\r\n            icon=\"trash-2\"\r\n            variant=\"warn\"\r\n            size=\"sm\"\r\n            ariaLabel=\"Elimina sezione\"\r\n            tooltip=\"Elimina sezione\"\r\n            (click)=\"sectionDelete.emit()\"\r\n          />\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Corpo della sezione (espandibile) -->\r\n      @if (isExpanded) {\r\n        <div class=\"section-body\">\r\n          <!-- Drop list CDK per i campi -->\r\n          <div\r\n            cdkDropList\r\n            [cdkDropListData]=\"section\"\r\n            [id]=\"'section-' + section.id\"\r\n            (cdkDropListDropped)=\"onFieldDrop($event)\"\r\n            class=\"field-drop-list\"\r\n          >\r\n            @for (field of section.fields; track field.key) {\r\n              <div\r\n                class=\"field-item\"\r\n                cdkDrag\r\n                [cdkDragData]=\"field\"\r\n                [class.field-item--selected]=\"field.key === selectedFieldKey\"\r\n                (click)=\"fieldSelect.emit(field.key)\"\r\n              >\r\n                <!-- Handle di trascinamento -->\r\n                <div class=\"field-item__handle\" cdkDragHandle>\r\n                  <lucide-icon name=\"grip-vertical\" [size]=\"16\" />\r\n                </div>\r\n\r\n                <!-- Icona tipo campo -->\r\n                <div class=\"field-item__icon\">\r\n                  <lucide-icon [name]=\"getFieldIcon(field.type)\" [size]=\"16\" />\r\n                </div>\r\n\r\n                <!-- Informazioni campo -->\r\n                <div class=\"field-item__info\">\r\n                  <span class=\"field-item__label\">\r\n                    {{ field.label || 'Senza etichetta' }}\r\n                    @if (field.required) {\r\n                      <span class=\"field-item__required\">*</span>\r\n                    }\r\n                  </span>\r\n                  <span class=\"field-item__key\">{{ field.key }}</span>\r\n                </div>\r\n\r\n                <!-- Badge tipo campo -->\r\n                <span class=\"field-item__type-badge\">{{ field.type }}</span>\r\n\r\n                <!-- Azioni campo (visibili al passaggio del mouse) -->\r\n                <div class=\"field-item__actions\" (click)=\"$event.stopPropagation()\">\r\n                  <ui-button\r\n                    icon=\"copy\"\r\n                    variant=\"ghost\"\r\n                    size=\"xs\"\r\n                    ariaLabel=\"Duplica campo\"\r\n                    tooltip=\"Duplica campo\"\r\n                    (click)=\"fieldDuplicate.emit(field.key)\"\r\n                  />\r\n\r\n                  <ui-button\r\n                    icon=\"trash-2\"\r\n                    variant=\"warn\"\r\n                    size=\"xs\"\r\n                    ariaLabel=\"Elimina campo\"\r\n                    tooltip=\"Elimina campo\"\r\n                    (click)=\"fieldDelete.emit(field.key)\"\r\n                  />\r\n                </div>\r\n              </div>\r\n            } @empty {\r\n              <!-- Stato vuoto -->\r\n              <div class=\"field-drop-list__empty\">\r\n                <lucide-icon name=\"mouse-pointer-click\" [size]=\"24\" />\r\n                <span>Trascina qui i campi dalla palette</span>\r\n              </div>\r\n            }\r\n          </div>\r\n        </div>\r\n      }\r\n    </mat-card>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* ─── Card sezione ──────────────────────────────────────── */\r\n\r\n    :host {\r\n      display: block;\r\n    }\r\n\r\n    .section-card {\r\n      border: 2px solid var(--ui-color-border);\r\n      border-radius: var(--ui-radius-md);\r\n      overflow: hidden;\r\n      transition: border-color var(--ui-transition-fast),\r\n                  box-shadow var(--ui-transition-fast);\r\n    }\r\n\r\n    .section-card:hover {\r\n      box-shadow: var(--ui-shadow-sm);\r\n    }\r\n\r\n    .section-card--selected {\r\n      border-color: var(--ui-color-primary);\r\n      box-shadow: var(--ui-shadow-md);\r\n    }\r\n\r\n    /* ─── Header sezione ────────────────────────────────────── */\r\n\r\n    .section-header {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n      padding: var(--ui-spacing-3) var(--ui-spacing-4);\r\n      background: var(--ui-color-bg-subtle);\r\n      cursor: pointer;\r\n      user-select: none;\r\n      transition: background var(--ui-transition-fast);\r\n    }\r\n\r\n    .section-header:hover {\r\n      background: var(--ui-color-surface);\r\n    }\r\n\r\n    .section-header__title {\r\n      flex: 1;\r\n      font-size: var(--ui-font-size-md);\r\n      font-weight: 600;\r\n      color: var(--ui-color-text);\r\n      cursor: pointer;\r\n    }\r\n\r\n    .section-header__title-field {\r\n      flex: 1;\r\n    }\r\n\r\n    /* Riduce il padding del form field inline */\r\n    .section-header__title-field ::ng-deep .mat-mdc-form-field-infix {\r\n      padding-top: var(--ui-spacing-1) !important;\r\n      padding-bottom: var(--ui-spacing-1) !important;\r\n      min-height: unset;\r\n    }\r\n\r\n    .section-header__badge {\r\n      font-size: var(--ui-font-size-xs);\r\n      color: var(--ui-color-text-muted);\r\n      background: var(--ui-color-surface);\r\n      padding: var(--ui-spacing-1) var(--ui-spacing-2);\r\n      border-radius: var(--ui-radius-sm);\r\n      white-space: nowrap;\r\n    }\r\n\r\n    .section-header__actions {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1);\r\n      opacity: 0;\r\n      transition: opacity var(--ui-transition-fast);\r\n    }\r\n\r\n    .section-header:hover .section-header__actions {\r\n      opacity: 1;\r\n    }\r\n\r\n    /* ─── Corpo sezione ─────────────────────────────────────── */\r\n\r\n    .section-body {\r\n      padding: var(--ui-spacing-3);\r\n    }\r\n\r\n    /* ─── Drop list campi ───────────────────────────────────── */\r\n\r\n    .field-drop-list {\r\n      min-height: 48px;\r\n      border: 2px dashed var(--ui-color-border);\r\n      border-radius: var(--ui-radius-sm);\r\n      transition: border-color var(--ui-transition-fast),\r\n                  background var(--ui-transition-fast);\r\n    }\r\n\r\n    .field-drop-list.cdk-drop-list-dragging {\r\n      border-color: var(--ui-color-primary);\r\n      background: var(--ui-color-primary-bg);\r\n    }\r\n\r\n    .field-drop-list__empty {\r\n      display: flex;\r\n      flex-direction: column;\r\n      align-items: center;\r\n      justify-content: center;\r\n      gap: var(--ui-spacing-2);\r\n      padding: var(--ui-spacing-6) var(--ui-spacing-4);\r\n      color: var(--ui-color-text-muted);\r\n      font-size: var(--ui-font-size-sm);\r\n    }\r\n\r\n    /* ─── Elemento campo ────────────────────────────────────── */\r\n\r\n    .field-item {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n      padding: var(--ui-spacing-2) var(--ui-spacing-3);\r\n      background: var(--ui-color-surface);\r\n      border-bottom: 1px solid var(--ui-color-border);\r\n      cursor: pointer;\r\n      transition: background var(--ui-transition-fast);\r\n    }\r\n\r\n    .field-item:first-child {\r\n      border-top-left-radius: var(--ui-radius-sm);\r\n      border-top-right-radius: var(--ui-radius-sm);\r\n    }\r\n\r\n    .field-item:last-child {\r\n      border-bottom: none;\r\n      border-bottom-left-radius: var(--ui-radius-sm);\r\n      border-bottom-right-radius: var(--ui-radius-sm);\r\n    }\r\n\r\n    .field-item:hover {\r\n      background: var(--ui-color-bg-subtle);\r\n    }\r\n\r\n    .field-item--selected {\r\n      background: var(--ui-color-primary-bg);\r\n      border-left: 3px solid var(--ui-color-primary);\r\n    }\r\n\r\n    /* Handle di trascinamento */\r\n    .field-item__handle {\r\n      display: flex;\r\n      align-items: center;\r\n      cursor: grab;\r\n      color: var(--ui-color-text-muted);\r\n      transition: color var(--ui-transition-fast);\r\n    }\r\n\r\n    .field-item__handle:active {\r\n      cursor: grabbing;\r\n    }\r\n\r\n    .field-item:hover .field-item__handle {\r\n      color: var(--ui-color-text-secondary);\r\n    }\r\n\r\n    /* Icona tipo campo */\r\n    .field-item__icon {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      width: 28px;\r\n      height: 28px;\r\n      border-radius: var(--ui-radius-sm);\r\n      background: var(--ui-color-bg-subtle);\r\n      color: var(--ui-color-primary);\r\n    }\r\n\r\n    /* Informazioni campo */\r\n    .field-item__info {\r\n      flex: 1;\r\n      display: flex;\r\n      flex-direction: column;\r\n      min-width: 0;\r\n    }\r\n\r\n    .field-item__label {\r\n      font-size: var(--ui-font-size-sm);\r\n      font-weight: 500;\r\n      color: var(--ui-color-text);\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n    }\r\n\r\n    .field-item__required {\r\n      color: var(--ui-color-warn);\r\n      margin-left: var(--ui-spacing-1);\r\n    }\r\n\r\n    .field-item__key {\r\n      font-size: var(--ui-font-size-xs);\r\n      color: var(--ui-color-text-muted);\r\n      font-family: monospace;\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n    }\r\n\r\n    /* Badge tipo campo */\r\n    .field-item__type-badge {\r\n      font-size: var(--ui-font-size-xs);\r\n      color: var(--ui-color-primary);\r\n      background: var(--ui-color-primary-bg);\r\n      padding: 2px var(--ui-spacing-2);\r\n      border-radius: var(--ui-radius-sm);\r\n      white-space: nowrap;\r\n      font-weight: 500;\r\n    }\r\n\r\n    /* Azioni campo */\r\n    .field-item__actions {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1);\r\n      opacity: 0;\r\n      transition: opacity var(--ui-transition-fast);\r\n    }\r\n\r\n    .field-item:hover .field-item__actions {\r\n      opacity: 1;\r\n    }\r\n\r\n    /* ─── CDK Drag & Drop ───────────────────────────────────── */\r\n\r\n    .cdk-drag-preview {\r\n      box-sizing: border-box;\r\n      border-radius: var(--ui-radius-sm);\r\n      box-shadow: var(--ui-shadow-md);\r\n      background: var(--ui-color-surface);\r\n      padding: var(--ui-spacing-2) var(--ui-spacing-3);\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n    }\r\n\r\n    .cdk-drag-placeholder {\r\n      opacity: 0.3;\r\n    }\r\n\r\n    .cdk-drag-animating {\r\n      transition: transform 250ms cubic-bezier(0, 0, 0.2, 1);\r\n    }\r\n  `,\r\n  ],\r\n})\r\nexport class UiSectionEditorComponent {\r\n  /** Sezione del form da renderizzare. */\r\n  @Input({ required: true }) section!: UiFormSection;\r\n\r\n  /** Indica se la sezione e attualmente selezionata. */\r\n  @Input() isSelected = false;\r\n\r\n  /** Chiave del campo attualmente selezionato (o null). */\r\n  @Input() selectedFieldKey: string | null = null;\r\n\r\n  /** Emette aggiornamenti parziali alla sezione. */\r\n  @Output() sectionUpdate = new EventEmitter<Partial<UiFormSection>>();\r\n\r\n  /** Emette richiesta di eliminazione della sezione. */\r\n  @Output() sectionDelete = new EventEmitter<void>();\r\n\r\n  /** Emette richiesta di duplicazione della sezione. */\r\n  @Output() sectionDuplicate = new EventEmitter<void>();\r\n\r\n  /** Emette la chiave del campo selezionato. */\r\n  @Output() fieldSelect = new EventEmitter<string>();\r\n\r\n  /** Emette la chiave del campo da duplicare. */\r\n  @Output() fieldDuplicate = new EventEmitter<string>();\r\n\r\n  /** Emette la chiave del campo da eliminare. */\r\n  @Output() fieldDelete = new EventEmitter<string>();\r\n\r\n  /** Emette l'evento di drop del CDK drag & drop. */\r\n  @Output() fieldDrop = new EventEmitter<CdkDragDrop<any>>();\r\n\r\n  /** Stato di espansione della sezione. */\r\n  isExpanded = true;\r\n\r\n  /** Indica se il titolo e in fase di modifica inline. */\r\n  isEditingTitle = false;\r\n\r\n  /** Titolo temporaneo durante la modifica inline. */\r\n  tempTitle = '';\r\n\r\n  /**\r\n   * Alterna lo stato di espansione/collasso della sezione.\r\n   */\r\n  toggleExpand(): void {\r\n    this.isExpanded = !this.isExpanded;\r\n  }\r\n\r\n  /**\r\n   * Avvia la modifica inline del titolo della sezione.\r\n   * @param event - Evento del mouse per prevenire la propagazione.\r\n   */\r\n  startTitleEdit(event: Event): void {\r\n    event.stopPropagation();\r\n    this.tempTitle = this.section.title || '';\r\n    this.isEditingTitle = true;\r\n  }\r\n\r\n  /**\r\n   * Conferma la modifica del titolo ed emette l'aggiornamento.\r\n   */\r\n  confirmTitleEdit(): void {\r\n    if (this.isEditingTitle) {\r\n      this.isEditingTitle = false;\r\n      const trimmed = this.tempTitle.trim();\r\n      if (trimmed && trimmed !== this.section.title) {\r\n        this.sectionUpdate.emit({ title: trimmed });\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Annulla la modifica del titolo, ripristinando il valore originale.\r\n   */\r\n  cancelTitleEdit(): void {\r\n    this.isEditingTitle = false;\r\n    this.tempTitle = '';\r\n  }\r\n\r\n  /**\r\n   * Restituisce il nome dell'icona Lucide associata al tipo di campo.\r\n   * @param type - Tipo di campo del form.\r\n   * @returns Nome dell'icona Lucide.\r\n   */\r\n  getFieldIcon(type: UiFieldType): string {\r\n    return FIELD_ICON_MAP[type] || 'help-circle';\r\n  }\r\n\r\n  /**\r\n   * Gestisce l'evento di drop del CDK drag & drop e lo propaga al parent.\r\n   * @param event - Evento CdkDragDrop.\r\n   */\r\n  onFieldDrop(event: CdkDragDrop<any>): void {\r\n    this.fieldDrop.emit(event);\r\n  }\r\n}\r\n"]}