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