@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,267 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy } from '@angular/core';
|
|
2
|
+
import { MatExpansionModule } from '@angular/material/expansion';
|
|
3
|
+
import { CdkDrag, CdkDragPlaceholder, CdkDropList } from '@angular/cdk/drag-drop';
|
|
4
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/material/expansion";
|
|
7
|
+
import * as i2 from "lucide-angular";
|
|
8
|
+
/**
|
|
9
|
+
* Palette laterale dei tipi di campo disponibili.
|
|
10
|
+
*
|
|
11
|
+
* Mostra i campi raggruppati per categoria tramite pannelli espandibili.
|
|
12
|
+
* Ogni elemento e trascinabile (CDK drag-drop) verso le sezioni del form.
|
|
13
|
+
*
|
|
14
|
+
* @selector ui-field-palette
|
|
15
|
+
*/
|
|
16
|
+
export class UiFieldPaletteComponent {
|
|
17
|
+
constructor() {
|
|
18
|
+
/**
|
|
19
|
+
* Predicato che impedisce il drop di elementi dentro la palette.
|
|
20
|
+
* La palette e solo una sorgente di drag, non una destinazione.
|
|
21
|
+
*/
|
|
22
|
+
this.noDrop = () => false;
|
|
23
|
+
/**
|
|
24
|
+
* Catalogo completo degli elementi della palette,
|
|
25
|
+
* organizzato per categorie con i rispettivi metadati.
|
|
26
|
+
*/
|
|
27
|
+
this.categories = [
|
|
28
|
+
{
|
|
29
|
+
key: 'basic',
|
|
30
|
+
label: 'Campi base',
|
|
31
|
+
icon: 'type',
|
|
32
|
+
items: [
|
|
33
|
+
{ type: 'text', label: 'Testo', icon: 'type', description: 'Campo di testo singola riga', category: 'basic' },
|
|
34
|
+
{ type: 'email', label: 'Email', icon: 'mail', description: 'Campo email con validazione', category: 'basic' },
|
|
35
|
+
{
|
|
36
|
+
type: 'password',
|
|
37
|
+
label: 'Password',
|
|
38
|
+
icon: 'lock',
|
|
39
|
+
description: 'Campo password mascherato',
|
|
40
|
+
category: 'basic',
|
|
41
|
+
},
|
|
42
|
+
{ type: 'number', label: 'Numero', icon: 'hash', description: 'Campo numerico', category: 'basic' },
|
|
43
|
+
{
|
|
44
|
+
type: 'textarea',
|
|
45
|
+
label: 'Area testo',
|
|
46
|
+
icon: 'text',
|
|
47
|
+
description: 'Campo di testo multilinea',
|
|
48
|
+
category: 'basic',
|
|
49
|
+
},
|
|
50
|
+
],
|
|
51
|
+
},
|
|
52
|
+
{
|
|
53
|
+
key: 'choice',
|
|
54
|
+
label: 'Campi scelta',
|
|
55
|
+
icon: 'list',
|
|
56
|
+
items: [
|
|
57
|
+
{
|
|
58
|
+
type: 'select',
|
|
59
|
+
label: 'Select',
|
|
60
|
+
icon: 'chevron-down',
|
|
61
|
+
description: 'Menu a tendina singola scelta',
|
|
62
|
+
category: 'choice',
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
type: 'multiselect',
|
|
66
|
+
label: 'Multiselect',
|
|
67
|
+
icon: 'list',
|
|
68
|
+
description: 'Selezione multipla',
|
|
69
|
+
category: 'choice',
|
|
70
|
+
},
|
|
71
|
+
{
|
|
72
|
+
type: 'freemultiselect',
|
|
73
|
+
label: 'Multiselect libera',
|
|
74
|
+
icon: 'list-plus',
|
|
75
|
+
description: 'Selezione multipla con valori liberi',
|
|
76
|
+
category: 'choice',
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
type: 'radio',
|
|
80
|
+
label: 'Radio',
|
|
81
|
+
icon: 'circle-dot',
|
|
82
|
+
description: 'Gruppo di opzioni radio',
|
|
83
|
+
category: 'choice',
|
|
84
|
+
},
|
|
85
|
+
{
|
|
86
|
+
type: 'checkbox',
|
|
87
|
+
label: 'Checkbox',
|
|
88
|
+
icon: 'check-square',
|
|
89
|
+
description: 'Casella di spunta',
|
|
90
|
+
category: 'choice',
|
|
91
|
+
},
|
|
92
|
+
{
|
|
93
|
+
type: 'switch',
|
|
94
|
+
label: 'Switch',
|
|
95
|
+
icon: 'toggle-left',
|
|
96
|
+
description: 'Interruttore on/off',
|
|
97
|
+
category: 'choice',
|
|
98
|
+
},
|
|
99
|
+
],
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
key: 'datetime',
|
|
103
|
+
label: 'Data e ora',
|
|
104
|
+
icon: 'calendar',
|
|
105
|
+
items: [
|
|
106
|
+
{ type: 'date', label: 'Data', icon: 'calendar', description: 'Selettore di data', category: 'datetime' },
|
|
107
|
+
{
|
|
108
|
+
type: 'datetime',
|
|
109
|
+
label: 'Data e ora',
|
|
110
|
+
icon: 'clock',
|
|
111
|
+
description: 'Selettore data e ora',
|
|
112
|
+
category: 'datetime',
|
|
113
|
+
},
|
|
114
|
+
],
|
|
115
|
+
},
|
|
116
|
+
{
|
|
117
|
+
key: 'advanced',
|
|
118
|
+
label: 'Avanzati',
|
|
119
|
+
icon: 'settings',
|
|
120
|
+
items: [
|
|
121
|
+
{ type: 'file', label: 'File', icon: 'upload', description: 'Caricamento file', category: 'advanced' },
|
|
122
|
+
{ type: 'flag', label: 'Flag', icon: 'flag', description: 'Campo flag booleano', category: 'advanced' },
|
|
123
|
+
],
|
|
124
|
+
},
|
|
125
|
+
{
|
|
126
|
+
key: 'custom',
|
|
127
|
+
label: 'Personalizzati',
|
|
128
|
+
icon: 'settings',
|
|
129
|
+
items: [
|
|
130
|
+
{
|
|
131
|
+
type: 'custom',
|
|
132
|
+
label: 'Custom',
|
|
133
|
+
icon: 'settings',
|
|
134
|
+
description: 'Componente personalizzato',
|
|
135
|
+
category: 'custom',
|
|
136
|
+
},
|
|
137
|
+
{ type: 'divider', label: 'Separatore', icon: 'minus', description: 'Divisore visuale', category: 'custom' },
|
|
138
|
+
],
|
|
139
|
+
},
|
|
140
|
+
];
|
|
141
|
+
}
|
|
142
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFieldPaletteComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
143
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiFieldPaletteComponent, isStandalone: true, selector: "ui-field-palette", ngImport: i0, template: `
|
|
144
|
+
<div class="field-palette">
|
|
145
|
+
<h3 class="field-palette__header">Campi disponibili</h3>
|
|
146
|
+
|
|
147
|
+
<mat-accordion multi>
|
|
148
|
+
@for (category of categories; track category.key) {
|
|
149
|
+
<mat-expansion-panel [expanded]="category.key === 'basic'">
|
|
150
|
+
<mat-expansion-panel-header>
|
|
151
|
+
<mat-panel-title>
|
|
152
|
+
<lucide-icon [name]="category.icon" [size]="16" aria-hidden="true" />
|
|
153
|
+
<span>{{ category.label }}</span>
|
|
154
|
+
</mat-panel-title>
|
|
155
|
+
</mat-expansion-panel-header>
|
|
156
|
+
|
|
157
|
+
<div
|
|
158
|
+
cdkDropList
|
|
159
|
+
[id]="'palette-' + category.key"
|
|
160
|
+
[cdkDropListData]="category.items"
|
|
161
|
+
[cdkDropListSortingDisabled]="true"
|
|
162
|
+
[cdkDropListEnterPredicate]="noDrop"
|
|
163
|
+
class="field-palette__list"
|
|
164
|
+
>
|
|
165
|
+
@for (item of category.items; track item.type) {
|
|
166
|
+
<div
|
|
167
|
+
cdkDrag
|
|
168
|
+
[cdkDragData]="item"
|
|
169
|
+
class="field-palette__item"
|
|
170
|
+
>
|
|
171
|
+
<!-- Placeholder visualizzato nella lista di destinazione durante il drag -->
|
|
172
|
+
<div class="field-palette__drag-placeholder" *cdkDragPlaceholder></div>
|
|
173
|
+
|
|
174
|
+
<!-- Icona handle di trascinamento (decorativa, l'intero elemento e draggabile) -->
|
|
175
|
+
<lucide-icon
|
|
176
|
+
name="grip-vertical"
|
|
177
|
+
[size]="14"
|
|
178
|
+
class="field-palette__drag-handle"
|
|
179
|
+
aria-hidden="true"
|
|
180
|
+
/>
|
|
181
|
+
|
|
182
|
+
<!-- Icona del tipo di campo -->
|
|
183
|
+
<lucide-icon
|
|
184
|
+
[name]="item.icon"
|
|
185
|
+
[size]="18"
|
|
186
|
+
class="field-palette__item-icon"
|
|
187
|
+
aria-hidden="true"
|
|
188
|
+
/>
|
|
189
|
+
|
|
190
|
+
<!-- Informazioni sul campo -->
|
|
191
|
+
<div class="field-palette__item-info">
|
|
192
|
+
<span class="field-palette__item-label">{{ item.label }}</span>
|
|
193
|
+
<span class="field-palette__item-desc">{{ item.description }}</span>
|
|
194
|
+
</div>
|
|
195
|
+
</div>
|
|
196
|
+
}
|
|
197
|
+
</div>
|
|
198
|
+
</mat-expansion-panel>
|
|
199
|
+
}
|
|
200
|
+
</mat-accordion>
|
|
201
|
+
</div>
|
|
202
|
+
`, isInline: true, styles: [".field-palette{padding:var(--ui-spacing-3, 12px);height:100%;overflow-y:auto;background:var(--ui-color-surface, #fff)}.field-palette__header{margin:0 0 var(--ui-spacing-3, 12px);font-size:var(--ui-font-size-md, 1rem);font-weight:600;color:var(--ui-color-text, #212121)}mat-panel-title{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);font-size:var(--ui-font-size-sm, .875rem);font-weight:500}.field-palette__list{display:flex;flex-direction:column;gap:var(--ui-spacing-1, 4px);min-height:24px}.field-palette__item{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-2, 8px);border:1px solid var(--ui-color-border, #e0e0e0);border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-surface, #fff);cursor:grab;transition:box-shadow var(--ui-transition-fast, .15s),border-color var(--ui-transition-fast, .15s);-webkit-user-select:none;user-select:none}.field-palette__item:hover{border-color:var(--ui-color-primary, #3f51b5);box-shadow:var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,.1))}.field-palette__drag-handle{color:var(--ui-color-text-muted, #9e9e9e);flex-shrink:0;cursor:grab}.field-palette__item-icon{color:var(--ui-color-primary, #3f51b5);flex-shrink:0}.field-palette__item-info{display:flex;flex-direction:column;min-width:0}.field-palette__item-label{font-size:var(--ui-font-size-sm, .875rem);font-weight:500;color:var(--ui-color-text, #212121);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-palette__item-desc{font-size:var(--ui-font-size-xs, .75rem);color:var(--ui-color-text-secondary, #616161);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-palette__drag-placeholder{height:40px;border:2px dashed var(--ui-color-primary, #3f51b5);border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-primary-bg, rgba(63, 81, 181, .08));transition:transform var(--ui-transition-fast, .15s)}.cdk-drag-preview{box-shadow:var(--ui-shadow-md, 0 4px 12px rgba(0,0,0,.15));border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-surface, #fff);padding:var(--ui-spacing-2, 8px);display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.cdk-drag-animating{transition:transform var(--ui-transition-normal, .25s) cubic-bezier(0,0,.2,1)}mat-expansion-panel{margin-bottom:var(--ui-spacing-1, 4px)!important;box-shadow:none!important;border:1px solid var(--ui-color-border, #e0e0e0);border-radius:var(--ui-radius-sm, 4px)!important}\n"], dependencies: [{ kind: "ngmodule", type: MatExpansionModule }, { kind: "directive", type: i1.MatAccordion, selector: "mat-accordion", inputs: ["hideToggle", "displayMode", "togglePosition"], exportAs: ["matAccordion"] }, { kind: "component", type: i1.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i1.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i1.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "directive", type: 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: 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: CdkDragPlaceholder, selector: "ng-template[cdkDragPlaceholder]", inputs: ["data"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
203
|
+
}
|
|
204
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFieldPaletteComponent, decorators: [{
|
|
205
|
+
type: Component,
|
|
206
|
+
args: [{ selector: 'ui-field-palette', standalone: true, imports: [MatExpansionModule, CdkDropList, CdkDrag, CdkDragPlaceholder, LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
207
|
+
<div class="field-palette">
|
|
208
|
+
<h3 class="field-palette__header">Campi disponibili</h3>
|
|
209
|
+
|
|
210
|
+
<mat-accordion multi>
|
|
211
|
+
@for (category of categories; track category.key) {
|
|
212
|
+
<mat-expansion-panel [expanded]="category.key === 'basic'">
|
|
213
|
+
<mat-expansion-panel-header>
|
|
214
|
+
<mat-panel-title>
|
|
215
|
+
<lucide-icon [name]="category.icon" [size]="16" aria-hidden="true" />
|
|
216
|
+
<span>{{ category.label }}</span>
|
|
217
|
+
</mat-panel-title>
|
|
218
|
+
</mat-expansion-panel-header>
|
|
219
|
+
|
|
220
|
+
<div
|
|
221
|
+
cdkDropList
|
|
222
|
+
[id]="'palette-' + category.key"
|
|
223
|
+
[cdkDropListData]="category.items"
|
|
224
|
+
[cdkDropListSortingDisabled]="true"
|
|
225
|
+
[cdkDropListEnterPredicate]="noDrop"
|
|
226
|
+
class="field-palette__list"
|
|
227
|
+
>
|
|
228
|
+
@for (item of category.items; track item.type) {
|
|
229
|
+
<div
|
|
230
|
+
cdkDrag
|
|
231
|
+
[cdkDragData]="item"
|
|
232
|
+
class="field-palette__item"
|
|
233
|
+
>
|
|
234
|
+
<!-- Placeholder visualizzato nella lista di destinazione durante il drag -->
|
|
235
|
+
<div class="field-palette__drag-placeholder" *cdkDragPlaceholder></div>
|
|
236
|
+
|
|
237
|
+
<!-- Icona handle di trascinamento (decorativa, l'intero elemento e draggabile) -->
|
|
238
|
+
<lucide-icon
|
|
239
|
+
name="grip-vertical"
|
|
240
|
+
[size]="14"
|
|
241
|
+
class="field-palette__drag-handle"
|
|
242
|
+
aria-hidden="true"
|
|
243
|
+
/>
|
|
244
|
+
|
|
245
|
+
<!-- Icona del tipo di campo -->
|
|
246
|
+
<lucide-icon
|
|
247
|
+
[name]="item.icon"
|
|
248
|
+
[size]="18"
|
|
249
|
+
class="field-palette__item-icon"
|
|
250
|
+
aria-hidden="true"
|
|
251
|
+
/>
|
|
252
|
+
|
|
253
|
+
<!-- Informazioni sul campo -->
|
|
254
|
+
<div class="field-palette__item-info">
|
|
255
|
+
<span class="field-palette__item-label">{{ item.label }}</span>
|
|
256
|
+
<span class="field-palette__item-desc">{{ item.description }}</span>
|
|
257
|
+
</div>
|
|
258
|
+
</div>
|
|
259
|
+
}
|
|
260
|
+
</div>
|
|
261
|
+
</mat-expansion-panel>
|
|
262
|
+
}
|
|
263
|
+
</mat-accordion>
|
|
264
|
+
</div>
|
|
265
|
+
`, styles: [".field-palette{padding:var(--ui-spacing-3, 12px);height:100%;overflow-y:auto;background:var(--ui-color-surface, #fff)}.field-palette__header{margin:0 0 var(--ui-spacing-3, 12px);font-size:var(--ui-font-size-md, 1rem);font-weight:600;color:var(--ui-color-text, #212121)}mat-panel-title{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);font-size:var(--ui-font-size-sm, .875rem);font-weight:500}.field-palette__list{display:flex;flex-direction:column;gap:var(--ui-spacing-1, 4px);min-height:24px}.field-palette__item{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-2, 8px);border:1px solid var(--ui-color-border, #e0e0e0);border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-surface, #fff);cursor:grab;transition:box-shadow var(--ui-transition-fast, .15s),border-color var(--ui-transition-fast, .15s);-webkit-user-select:none;user-select:none}.field-palette__item:hover{border-color:var(--ui-color-primary, #3f51b5);box-shadow:var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,.1))}.field-palette__drag-handle{color:var(--ui-color-text-muted, #9e9e9e);flex-shrink:0;cursor:grab}.field-palette__item-icon{color:var(--ui-color-primary, #3f51b5);flex-shrink:0}.field-palette__item-info{display:flex;flex-direction:column;min-width:0}.field-palette__item-label{font-size:var(--ui-font-size-sm, .875rem);font-weight:500;color:var(--ui-color-text, #212121);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-palette__item-desc{font-size:var(--ui-font-size-xs, .75rem);color:var(--ui-color-text-secondary, #616161);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.field-palette__drag-placeholder{height:40px;border:2px dashed var(--ui-color-primary, #3f51b5);border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-primary-bg, rgba(63, 81, 181, .08));transition:transform var(--ui-transition-fast, .15s)}.cdk-drag-preview{box-shadow:var(--ui-shadow-md, 0 4px 12px rgba(0,0,0,.15));border-radius:var(--ui-radius-sm, 4px);background:var(--ui-color-surface, #fff);padding:var(--ui-spacing-2, 8px);display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.cdk-drag-animating{transition:transform var(--ui-transition-normal, .25s) cubic-bezier(0,0,.2,1)}mat-expansion-panel{margin-bottom:var(--ui-spacing-1, 4px)!important;box-shadow:none!important;border:1px solid var(--ui-color-border, #e0e0e0);border-radius:var(--ui-radius-sm, 4px)!important}\n"] }]
|
|
266
|
+
}] });
|
|
267
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"field-palette.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ChangeDetectionStrategy, Input } from '@angular/core';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { CdkDrag, CdkDragPlaceholder, CdkDropList } from '@angular/cdk/drag-drop';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\n\r\nimport { UiEditorFieldPaletteItem, UiEditorFieldCategory } from '../../types/editor.types';\r\nimport { UiFieldType } from '../../../form-builder/types/index';\r\nimport { UiIconName } from '../../../../core/types';\r\n\r\n/**\r\n * Interfaccia interna per una categoria della palette.\r\n * Raggruppa un insieme di elementi sotto un'etichetta comune.\r\n */\r\ninterface PaletteCategory {\r\n  /** Identificativo della categoria. */\r\n  key: UiEditorFieldCategory;\r\n  /** Etichetta visualizzata nell'intestazione dell'accordion. */\r\n  label: string;\r\n  /** Icona Lucide della categoria. */\r\n  icon: UiIconName;\r\n  /** Elementi trascinabili appartenenti alla categoria. */\r\n  items: UiEditorFieldPaletteItem[];\r\n}\r\n\r\n/**\r\n * Palette laterale dei tipi di campo disponibili.\r\n *\r\n * Mostra i campi raggruppati per categoria tramite pannelli espandibili.\r\n * Ogni elemento e trascinabile (CDK drag-drop) verso le sezioni del form.\r\n *\r\n * @selector ui-field-palette\r\n */\r\n@Component({\r\n  selector: 'ui-field-palette',\r\n  standalone: true,\r\n  imports: [MatExpansionModule, CdkDropList, CdkDrag, CdkDragPlaceholder, LucideAngularModule],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"field-palette\">\r\n      <h3 class=\"field-palette__header\">Campi disponibili</h3>\r\n\r\n      <mat-accordion multi>\r\n        @for (category of categories; track category.key) {\r\n          <mat-expansion-panel [expanded]=\"category.key === 'basic'\">\r\n            <mat-expansion-panel-header>\r\n              <mat-panel-title>\r\n                <lucide-icon [name]=\"category.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                <span>{{ category.label }}</span>\r\n              </mat-panel-title>\r\n            </mat-expansion-panel-header>\r\n\r\n            <div\r\n              cdkDropList\r\n              [id]=\"'palette-' + category.key\"\r\n              [cdkDropListData]=\"category.items\"\r\n              [cdkDropListSortingDisabled]=\"true\"\r\n              [cdkDropListEnterPredicate]=\"noDrop\"\r\n              class=\"field-palette__list\"\r\n            >\r\n              @for (item of category.items; track item.type) {\r\n                <div\r\n                  cdkDrag\r\n                  [cdkDragData]=\"item\"\r\n                  class=\"field-palette__item\"\r\n                >\r\n                  <!-- Placeholder visualizzato nella lista di destinazione durante il drag -->\r\n                  <div class=\"field-palette__drag-placeholder\" *cdkDragPlaceholder></div>\r\n\r\n                  <!-- Icona handle di trascinamento (decorativa, l'intero elemento e draggabile) -->\r\n                  <lucide-icon\r\n                    name=\"grip-vertical\"\r\n                    [size]=\"14\"\r\n                    class=\"field-palette__drag-handle\"\r\n                    aria-hidden=\"true\"\r\n                  />\r\n\r\n                  <!-- Icona del tipo di campo -->\r\n                  <lucide-icon\r\n                    [name]=\"item.icon\"\r\n                    [size]=\"18\"\r\n                    class=\"field-palette__item-icon\"\r\n                    aria-hidden=\"true\"\r\n                  />\r\n\r\n                  <!-- Informazioni sul campo -->\r\n                  <div class=\"field-palette__item-info\">\r\n                    <span class=\"field-palette__item-label\">{{ item.label }}</span>\r\n                    <span class=\"field-palette__item-desc\">{{ item.description }}</span>\r\n                  </div>\r\n                </div>\r\n              }\r\n            </div>\r\n          </mat-expansion-panel>\r\n        }\r\n      </mat-accordion>\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* Contenitore principale della palette */\r\n    .field-palette {\r\n      padding: var(--ui-spacing-3, 12px);\r\n      height: 100%;\r\n      overflow-y: auto;\r\n      background: var(--ui-color-surface, #fff);\r\n    }\r\n\r\n    /* Intestazione della palette */\r\n    .field-palette__header {\r\n      margin: 0 0 var(--ui-spacing-3, 12px);\r\n      font-size: var(--ui-font-size-md, 1rem);\r\n      font-weight: 600;\r\n      color: var(--ui-color-text, #212121);\r\n    }\r\n\r\n    /* Titolo del pannello: icona + testo */\r\n    mat-panel-title {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      font-weight: 500;\r\n    }\r\n\r\n    /* Lista di campi all'interno del pannello */\r\n    .field-palette__list {\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: var(--ui-spacing-1, 4px);\r\n      min-height: 24px;\r\n    }\r\n\r\n    /* Singolo elemento trascinabile */\r\n    .field-palette__item {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n      padding: var(--ui-spacing-2, 8px);\r\n      border: 1px solid var(--ui-color-border, #e0e0e0);\r\n      border-radius: var(--ui-radius-sm, 4px);\r\n      background: var(--ui-color-surface, #fff);\r\n      cursor: grab;\r\n      transition: box-shadow var(--ui-transition-fast, 150ms),\r\n                  border-color var(--ui-transition-fast, 150ms);\r\n      user-select: none;\r\n    }\r\n\r\n    .field-palette__item:hover {\r\n      border-color: var(--ui-color-primary, #3f51b5);\r\n      box-shadow: var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,.1));\r\n    }\r\n\r\n    /* Handle di trascinamento */\r\n    .field-palette__drag-handle {\r\n      color: var(--ui-color-text-muted, #9e9e9e);\r\n      flex-shrink: 0;\r\n      cursor: grab;\r\n    }\r\n\r\n    /* Icona tipo campo */\r\n    .field-palette__item-icon {\r\n      color: var(--ui-color-primary, #3f51b5);\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    /* Blocco info (label + descrizione) */\r\n    .field-palette__item-info {\r\n      display: flex;\r\n      flex-direction: column;\r\n      min-width: 0;\r\n    }\r\n\r\n    .field-palette__item-label {\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      font-weight: 500;\r\n      color: var(--ui-color-text, #212121);\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n    }\r\n\r\n    .field-palette__item-desc {\r\n      font-size: var(--ui-font-size-xs, 0.75rem);\r\n      color: var(--ui-color-text-secondary, #616161);\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n    }\r\n\r\n    /* Placeholder mostrato nella zona di destinazione */\r\n    .field-palette__drag-placeholder {\r\n      height: 40px;\r\n      border: 2px dashed var(--ui-color-primary, #3f51b5);\r\n      border-radius: var(--ui-radius-sm, 4px);\r\n      background: var(--ui-color-primary-bg, rgba(63, 81, 181, 0.08));\r\n      transition: transform var(--ui-transition-fast, 150ms);\r\n    }\r\n\r\n    /* Stile dell'elemento durante il trascinamento (CDK) */\r\n    .cdk-drag-preview {\r\n      box-shadow: var(--ui-shadow-md, 0 4px 12px rgba(0,0,0,.15));\r\n      border-radius: var(--ui-radius-sm, 4px);\r\n      background: var(--ui-color-surface, #fff);\r\n      padding: var(--ui-spacing-2, 8px);\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n    }\r\n\r\n    /* Animazione di spostamento degli elementi CDK */\r\n    .cdk-drag-animating {\r\n      transition: transform var(--ui-transition-normal, 250ms) cubic-bezier(0, 0, 0.2, 1);\r\n    }\r\n\r\n    /* Pannelli espandibili: spaziatura */\r\n    mat-expansion-panel {\r\n      margin-bottom: var(--ui-spacing-1, 4px) !important;\r\n      box-shadow: none !important;\r\n      border: 1px solid var(--ui-color-border, #e0e0e0);\r\n      border-radius: var(--ui-radius-sm, 4px) !important;\r\n    }\r\n  `,\r\n  ],\r\n})\r\nexport class UiFieldPaletteComponent {\r\n  /**\r\n   * Predicato che impedisce il drop di elementi dentro la palette.\r\n   * La palette e solo una sorgente di drag, non una destinazione.\r\n   */\r\n  noDrop = (): boolean => false;\r\n\r\n  /**\r\n   * Catalogo completo degli elementi della palette,\r\n   * organizzato per categorie con i rispettivi metadati.\r\n   */\r\n  readonly categories: PaletteCategory[] = [\r\n    {\r\n      key: 'basic',\r\n      label: 'Campi base',\r\n      icon: 'type',\r\n      items: [\r\n        { type: 'text', label: 'Testo', icon: 'type', description: 'Campo di testo singola riga', category: 'basic' },\r\n        { type: 'email', label: 'Email', icon: 'mail', description: 'Campo email con validazione', category: 'basic' },\r\n        {\r\n          type: 'password',\r\n          label: 'Password',\r\n          icon: 'lock',\r\n          description: 'Campo password mascherato',\r\n          category: 'basic',\r\n        },\r\n        { type: 'number', label: 'Numero', icon: 'hash', description: 'Campo numerico', category: 'basic' },\r\n        {\r\n          type: 'textarea',\r\n          label: 'Area testo',\r\n          icon: 'text',\r\n          description: 'Campo di testo multilinea',\r\n          category: 'basic',\r\n        },\r\n      ],\r\n    },\r\n    {\r\n      key: 'choice',\r\n      label: 'Campi scelta',\r\n      icon: 'list',\r\n      items: [\r\n        {\r\n          type: 'select',\r\n          label: 'Select',\r\n          icon: 'chevron-down',\r\n          description: 'Menu a tendina singola scelta',\r\n          category: 'choice',\r\n        },\r\n        {\r\n          type: 'multiselect',\r\n          label: 'Multiselect',\r\n          icon: 'list',\r\n          description: 'Selezione multipla',\r\n          category: 'choice',\r\n        },\r\n        {\r\n          type: 'freemultiselect',\r\n          label: 'Multiselect libera',\r\n          icon: 'list-plus',\r\n          description: 'Selezione multipla con valori liberi',\r\n          category: 'choice',\r\n        },\r\n        {\r\n          type: 'radio',\r\n          label: 'Radio',\r\n          icon: 'circle-dot',\r\n          description: 'Gruppo di opzioni radio',\r\n          category: 'choice',\r\n        },\r\n        {\r\n          type: 'checkbox',\r\n          label: 'Checkbox',\r\n          icon: 'check-square',\r\n          description: 'Casella di spunta',\r\n          category: 'choice',\r\n        },\r\n        {\r\n          type: 'switch',\r\n          label: 'Switch',\r\n          icon: 'toggle-left',\r\n          description: 'Interruttore on/off',\r\n          category: 'choice',\r\n        },\r\n      ],\r\n    },\r\n    {\r\n      key: 'datetime',\r\n      label: 'Data e ora',\r\n      icon: 'calendar',\r\n      items: [\r\n        { type: 'date', label: 'Data', icon: 'calendar', description: 'Selettore di data', category: 'datetime' },\r\n        {\r\n          type: 'datetime',\r\n          label: 'Data e ora',\r\n          icon: 'clock',\r\n          description: 'Selettore data e ora',\r\n          category: 'datetime',\r\n        },\r\n      ],\r\n    },\r\n    {\r\n      key: 'advanced',\r\n      label: 'Avanzati',\r\n      icon: 'settings',\r\n      items: [\r\n        { type: 'file', label: 'File', icon: 'upload', description: 'Caricamento file', category: 'advanced' },\r\n        { type: 'flag', label: 'Flag', icon: 'flag', description: 'Campo flag booleano', category: 'advanced' },\r\n      ],\r\n    },\r\n    {\r\n      key: 'custom',\r\n      label: 'Personalizzati',\r\n      icon: 'settings',\r\n      items: [\r\n        {\r\n          type: 'custom',\r\n          label: 'Custom',\r\n          icon: 'settings',\r\n          description: 'Componente personalizzato',\r\n          category: 'custom',\r\n        },\r\n        { type: 'divider', label: 'Separatore', icon: 'minus', description: 'Divisore visuale', category: 'custom' },\r\n      ],\r\n    },\r\n  ];\r\n}\r\n"]}
|
|
@@ -0,0 +1,276 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { FormsModule } from '@angular/forms';
|
|
3
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
4
|
+
import { MatInputModule } from '@angular/material/input';
|
|
5
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
6
|
+
import { UiButtonComponent } from '../../../button/index';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "@angular/forms";
|
|
9
|
+
import * as i2 from "@angular/material/form-field";
|
|
10
|
+
import * as i3 from "@angular/material/input";
|
|
11
|
+
import * as i4 from "lucide-angular";
|
|
12
|
+
/**
|
|
13
|
+
* Pannello overlay in basso nell'anteprima che mostra i valori correnti
|
|
14
|
+
* del form in formato JSON modificabile.
|
|
15
|
+
*
|
|
16
|
+
* Permette di visualizzare, modificare, formattare e resettare i valori
|
|
17
|
+
* del form durante la fase di anteprima nell'editor.
|
|
18
|
+
*
|
|
19
|
+
* @selector ui-editor-form-values-panel
|
|
20
|
+
*/
|
|
21
|
+
export class UiEditorFormValuesPanelComponent {
|
|
22
|
+
constructor() {
|
|
23
|
+
/** Stato di apertura/chiusura del pannello. */
|
|
24
|
+
this.isOpen = false;
|
|
25
|
+
/** Emesso quando i valori vengono modificati dall'utente via JSON. */
|
|
26
|
+
this.valuesChange = new EventEmitter();
|
|
27
|
+
/** Emesso alla richiesta di aggiornamento dei valori. */
|
|
28
|
+
this.refreshValues = new EventEmitter();
|
|
29
|
+
/** Emesso alla richiesta di apertura/chiusura del pannello. */
|
|
30
|
+
this.toggle = new EventEmitter();
|
|
31
|
+
/** Testo JSON visualizzato nella textarea. */
|
|
32
|
+
this.jsonText = '{}';
|
|
33
|
+
/** Messaggio di errore per JSON non valido. */
|
|
34
|
+
this.jsonError = null;
|
|
35
|
+
/** @internal Valori interni del form. */
|
|
36
|
+
this._formValues = {};
|
|
37
|
+
}
|
|
38
|
+
/** Valori correnti del form da visualizzare. */
|
|
39
|
+
set formValues(value) {
|
|
40
|
+
this._formValues = value;
|
|
41
|
+
this.updateJsonText();
|
|
42
|
+
}
|
|
43
|
+
get formValues() {
|
|
44
|
+
return this._formValues;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Gestisce l'input dell'utente nella textarea JSON.
|
|
48
|
+
* Prova a parsare il JSON e aggiorna i valori se valido.
|
|
49
|
+
* @param text - Testo JSON inserito dall'utente
|
|
50
|
+
*/
|
|
51
|
+
onJsonInput(text) {
|
|
52
|
+
this.jsonText = text;
|
|
53
|
+
try {
|
|
54
|
+
const parsed = JSON.parse(text);
|
|
55
|
+
this.jsonError = null;
|
|
56
|
+
this.valuesChange.emit(parsed);
|
|
57
|
+
}
|
|
58
|
+
catch (e) {
|
|
59
|
+
this.jsonError = 'JSON non valido: controlla la sintassi';
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Pulisce i valori del form resettando a oggetto vuoto.
|
|
64
|
+
*/
|
|
65
|
+
clearValues() {
|
|
66
|
+
this._formValues = {};
|
|
67
|
+
this.jsonText = '{}';
|
|
68
|
+
this.jsonError = null;
|
|
69
|
+
this.valuesChange.emit({});
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Formatta il JSON con indentazione per una migliore leggibilita.
|
|
73
|
+
*/
|
|
74
|
+
formatJson() {
|
|
75
|
+
try {
|
|
76
|
+
const parsed = JSON.parse(this.jsonText);
|
|
77
|
+
this.jsonText = JSON.stringify(parsed, null, 2);
|
|
78
|
+
this.jsonError = null;
|
|
79
|
+
}
|
|
80
|
+
catch (e) {
|
|
81
|
+
this.jsonError = 'Impossibile formattare: JSON non valido';
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* @internal
|
|
86
|
+
* Aggiorna il testo JSON dalla rappresentazione interna dei valori.
|
|
87
|
+
*/
|
|
88
|
+
updateJsonText() {
|
|
89
|
+
try {
|
|
90
|
+
this.jsonText = JSON.stringify(this._formValues, null, 2);
|
|
91
|
+
this.jsonError = null;
|
|
92
|
+
}
|
|
93
|
+
catch {
|
|
94
|
+
this.jsonText = '{}';
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFormValuesPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
98
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiEditorFormValuesPanelComponent, isStandalone: true, selector: "ui-editor-form-values-panel", inputs: { formValues: "formValues", isOpen: "isOpen" }, outputs: { valuesChange: "valuesChange", refreshValues: "refreshValues", toggle: "toggle" }, ngImport: i0, template: `
|
|
99
|
+
<div
|
|
100
|
+
class="form-values-panel"
|
|
101
|
+
[class.form-values-panel--open]="isOpen"
|
|
102
|
+
>
|
|
103
|
+
<!-- Intestazione con toggle e azioni -->
|
|
104
|
+
<div class="form-values-panel__header" (click)="toggle.emit()">
|
|
105
|
+
<div class="form-values-panel__header-left">
|
|
106
|
+
<lucide-icon name="code" [size]="16" aria-hidden="true" />
|
|
107
|
+
<span class="form-values-panel__title">Valori Form</span>
|
|
108
|
+
</div>
|
|
109
|
+
|
|
110
|
+
<div class="form-values-panel__header-actions" (click)="$event.stopPropagation()">
|
|
111
|
+
<!-- Aggiorna valori -->
|
|
112
|
+
<ui-button
|
|
113
|
+
icon="refresh-cw"
|
|
114
|
+
variant="ghost"
|
|
115
|
+
size="xs"
|
|
116
|
+
tooltip="Aggiorna valori"
|
|
117
|
+
ariaLabel="Aggiorna valori"
|
|
118
|
+
(click)="refreshValues.emit()"
|
|
119
|
+
/>
|
|
120
|
+
|
|
121
|
+
<!-- Pulisci valori -->
|
|
122
|
+
<ui-button
|
|
123
|
+
icon="trash-2"
|
|
124
|
+
variant="ghost"
|
|
125
|
+
size="xs"
|
|
126
|
+
tooltip="Pulisci valori"
|
|
127
|
+
ariaLabel="Pulisci valori"
|
|
128
|
+
(click)="clearValues()"
|
|
129
|
+
/>
|
|
130
|
+
|
|
131
|
+
<!-- Formatta JSON -->
|
|
132
|
+
<ui-button
|
|
133
|
+
icon="align-left"
|
|
134
|
+
variant="ghost"
|
|
135
|
+
size="xs"
|
|
136
|
+
tooltip="Formatta JSON"
|
|
137
|
+
ariaLabel="Formatta JSON"
|
|
138
|
+
(click)="formatJson()"
|
|
139
|
+
/>
|
|
140
|
+
|
|
141
|
+
<!-- Freccia toggle apertura/chiusura -->
|
|
142
|
+
<ui-button
|
|
143
|
+
[icon]="isOpen ? 'chevron-down' : 'chevron-up'"
|
|
144
|
+
variant="ghost"
|
|
145
|
+
size="xs"
|
|
146
|
+
tooltip="Chiudi pannello"
|
|
147
|
+
ariaLabel="Chiudi pannello"
|
|
148
|
+
(click)="toggle.emit()"
|
|
149
|
+
/>
|
|
150
|
+
</div>
|
|
151
|
+
</div>
|
|
152
|
+
|
|
153
|
+
<!-- Corpo del pannello (visibile solo quando aperto) -->
|
|
154
|
+
@if (isOpen) {
|
|
155
|
+
<div class="form-values-panel__body">
|
|
156
|
+
<mat-form-field appearance="outline" class="form-values-panel__textarea-wrapper">
|
|
157
|
+
<mat-label>JSON</mat-label>
|
|
158
|
+
<textarea
|
|
159
|
+
matInput
|
|
160
|
+
class="form-values-panel__textarea"
|
|
161
|
+
[ngModel]="jsonText"
|
|
162
|
+
(ngModelChange)="onJsonInput($event)"
|
|
163
|
+
rows="8"
|
|
164
|
+
spellcheck="false"
|
|
165
|
+
placeholder='{ "chiave": "valore" }'
|
|
166
|
+
></textarea>
|
|
167
|
+
</mat-form-field>
|
|
168
|
+
|
|
169
|
+
<!-- Indicatore di errore di parsing -->
|
|
170
|
+
@if (jsonError) {
|
|
171
|
+
<div class="form-values-panel__error">
|
|
172
|
+
<lucide-icon name="alert-triangle" [size]="14" aria-hidden="true" />
|
|
173
|
+
<span>{{ jsonError }}</span>
|
|
174
|
+
</div>
|
|
175
|
+
}
|
|
176
|
+
</div>
|
|
177
|
+
}
|
|
178
|
+
</div>
|
|
179
|
+
`, isInline: true, styles: [".form-values-panel{border-top:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-surface, #fff);transition:max-height .2s ease-in-out;overflow:hidden}.form-values-panel__header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2, 8px) var(--ui-spacing-3, 12px);cursor:pointer;-webkit-user-select:none;user-select:none;transition:background-color .15s}.form-values-panel__header:hover{background:var(--ui-color-bg-subtle, #fafafa)}.form-values-panel__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.form-values-panel__title{font-weight:600;font-size:var(--ui-font-size-xs, .75rem);color:var(--ui-color-text-secondary, #616161);text-transform:uppercase;letter-spacing:.5px}.form-values-panel__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px)}.form-values-panel__body{padding:0 var(--ui-spacing-3, 12px) var(--ui-spacing-3, 12px)}.form-values-panel__textarea-wrapper{width:100%}.form-values-panel__textarea{font-family:Fira Code,Consolas,Monaco,monospace;font-size:var(--ui-font-size-xs, .75rem);line-height:1.5;resize:vertical}.form-values-panel__error{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px);padding:var(--ui-spacing-1, 4px) var(--ui-spacing-2, 8px);margin-top:var(--ui-spacing-1, 4px);border-radius:var(--ui-radius-sm, 4px);background:color-mix(in srgb,var(--ui-color-warn, #f44336) 10%,transparent);color:var(--ui-color-warn, #f44336);font-size:var(--ui-font-size-xs, .75rem)}\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: MatFormFieldModule }, { kind: "component", type: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.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: LucideAngularModule }, { kind: "component", type: i4.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
180
|
+
}
|
|
181
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFormValuesPanelComponent, decorators: [{
|
|
182
|
+
type: Component,
|
|
183
|
+
args: [{ selector: 'ui-editor-form-values-panel', standalone: true, imports: [FormsModule, MatFormFieldModule, MatInputModule, LucideAngularModule, UiButtonComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
184
|
+
<div
|
|
185
|
+
class="form-values-panel"
|
|
186
|
+
[class.form-values-panel--open]="isOpen"
|
|
187
|
+
>
|
|
188
|
+
<!-- Intestazione con toggle e azioni -->
|
|
189
|
+
<div class="form-values-panel__header" (click)="toggle.emit()">
|
|
190
|
+
<div class="form-values-panel__header-left">
|
|
191
|
+
<lucide-icon name="code" [size]="16" aria-hidden="true" />
|
|
192
|
+
<span class="form-values-panel__title">Valori Form</span>
|
|
193
|
+
</div>
|
|
194
|
+
|
|
195
|
+
<div class="form-values-panel__header-actions" (click)="$event.stopPropagation()">
|
|
196
|
+
<!-- Aggiorna valori -->
|
|
197
|
+
<ui-button
|
|
198
|
+
icon="refresh-cw"
|
|
199
|
+
variant="ghost"
|
|
200
|
+
size="xs"
|
|
201
|
+
tooltip="Aggiorna valori"
|
|
202
|
+
ariaLabel="Aggiorna valori"
|
|
203
|
+
(click)="refreshValues.emit()"
|
|
204
|
+
/>
|
|
205
|
+
|
|
206
|
+
<!-- Pulisci valori -->
|
|
207
|
+
<ui-button
|
|
208
|
+
icon="trash-2"
|
|
209
|
+
variant="ghost"
|
|
210
|
+
size="xs"
|
|
211
|
+
tooltip="Pulisci valori"
|
|
212
|
+
ariaLabel="Pulisci valori"
|
|
213
|
+
(click)="clearValues()"
|
|
214
|
+
/>
|
|
215
|
+
|
|
216
|
+
<!-- Formatta JSON -->
|
|
217
|
+
<ui-button
|
|
218
|
+
icon="align-left"
|
|
219
|
+
variant="ghost"
|
|
220
|
+
size="xs"
|
|
221
|
+
tooltip="Formatta JSON"
|
|
222
|
+
ariaLabel="Formatta JSON"
|
|
223
|
+
(click)="formatJson()"
|
|
224
|
+
/>
|
|
225
|
+
|
|
226
|
+
<!-- Freccia toggle apertura/chiusura -->
|
|
227
|
+
<ui-button
|
|
228
|
+
[icon]="isOpen ? 'chevron-down' : 'chevron-up'"
|
|
229
|
+
variant="ghost"
|
|
230
|
+
size="xs"
|
|
231
|
+
tooltip="Chiudi pannello"
|
|
232
|
+
ariaLabel="Chiudi pannello"
|
|
233
|
+
(click)="toggle.emit()"
|
|
234
|
+
/>
|
|
235
|
+
</div>
|
|
236
|
+
</div>
|
|
237
|
+
|
|
238
|
+
<!-- Corpo del pannello (visibile solo quando aperto) -->
|
|
239
|
+
@if (isOpen) {
|
|
240
|
+
<div class="form-values-panel__body">
|
|
241
|
+
<mat-form-field appearance="outline" class="form-values-panel__textarea-wrapper">
|
|
242
|
+
<mat-label>JSON</mat-label>
|
|
243
|
+
<textarea
|
|
244
|
+
matInput
|
|
245
|
+
class="form-values-panel__textarea"
|
|
246
|
+
[ngModel]="jsonText"
|
|
247
|
+
(ngModelChange)="onJsonInput($event)"
|
|
248
|
+
rows="8"
|
|
249
|
+
spellcheck="false"
|
|
250
|
+
placeholder='{ "chiave": "valore" }'
|
|
251
|
+
></textarea>
|
|
252
|
+
</mat-form-field>
|
|
253
|
+
|
|
254
|
+
<!-- Indicatore di errore di parsing -->
|
|
255
|
+
@if (jsonError) {
|
|
256
|
+
<div class="form-values-panel__error">
|
|
257
|
+
<lucide-icon name="alert-triangle" [size]="14" aria-hidden="true" />
|
|
258
|
+
<span>{{ jsonError }}</span>
|
|
259
|
+
</div>
|
|
260
|
+
}
|
|
261
|
+
</div>
|
|
262
|
+
}
|
|
263
|
+
</div>
|
|
264
|
+
`, styles: [".form-values-panel{border-top:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-surface, #fff);transition:max-height .2s ease-in-out;overflow:hidden}.form-values-panel__header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2, 8px) var(--ui-spacing-3, 12px);cursor:pointer;-webkit-user-select:none;user-select:none;transition:background-color .15s}.form-values-panel__header:hover{background:var(--ui-color-bg-subtle, #fafafa)}.form-values-panel__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.form-values-panel__title{font-weight:600;font-size:var(--ui-font-size-xs, .75rem);color:var(--ui-color-text-secondary, #616161);text-transform:uppercase;letter-spacing:.5px}.form-values-panel__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px)}.form-values-panel__body{padding:0 var(--ui-spacing-3, 12px) var(--ui-spacing-3, 12px)}.form-values-panel__textarea-wrapper{width:100%}.form-values-panel__textarea{font-family:Fira Code,Consolas,Monaco,monospace;font-size:var(--ui-font-size-xs, .75rem);line-height:1.5;resize:vertical}.form-values-panel__error{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px);padding:var(--ui-spacing-1, 4px) var(--ui-spacing-2, 8px);margin-top:var(--ui-spacing-1, 4px);border-radius:var(--ui-radius-sm, 4px);background:color-mix(in srgb,var(--ui-color-warn, #f44336) 10%,transparent);color:var(--ui-color-warn, #f44336);font-size:var(--ui-font-size-xs, .75rem)}\n"] }]
|
|
265
|
+
}], propDecorators: { formValues: [{
|
|
266
|
+
type: Input
|
|
267
|
+
}], isOpen: [{
|
|
268
|
+
type: Input
|
|
269
|
+
}], valuesChange: [{
|
|
270
|
+
type: Output
|
|
271
|
+
}], refreshValues: [{
|
|
272
|
+
type: Output
|
|
273
|
+
}], toggle: [{
|
|
274
|
+
type: Output
|
|
275
|
+
}] } });
|
|
276
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-values-panel.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter } from '@angular/core';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiButtonComponent } from '../../../button/index';\r\n\r\n/**\r\n * Pannello overlay in basso nell'anteprima che mostra i valori correnti\r\n * del form in formato JSON modificabile.\r\n *\r\n * Permette di visualizzare, modificare, formattare e resettare i valori\r\n * del form durante la fase di anteprima nell'editor.\r\n *\r\n * @selector ui-editor-form-values-panel\r\n */\r\n@Component({\r\n  selector: 'ui-editor-form-values-panel',\r\n  standalone: true,\r\n  imports: [FormsModule, MatFormFieldModule, MatInputModule, LucideAngularModule, UiButtonComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div\r\n      class=\"form-values-panel\"\r\n      [class.form-values-panel--open]=\"isOpen\"\r\n    >\r\n      <!-- Intestazione con toggle e azioni -->\r\n      <div class=\"form-values-panel__header\" (click)=\"toggle.emit()\">\r\n        <div class=\"form-values-panel__header-left\">\r\n          <lucide-icon name=\"code\" [size]=\"16\" aria-hidden=\"true\" />\r\n          <span class=\"form-values-panel__title\">Valori Form</span>\r\n        </div>\r\n\r\n        <div class=\"form-values-panel__header-actions\" (click)=\"$event.stopPropagation()\">\r\n          <!-- Aggiorna valori -->\r\n          <ui-button\r\n            icon=\"refresh-cw\"\r\n            variant=\"ghost\"\r\n            size=\"xs\"\r\n            tooltip=\"Aggiorna valori\"\r\n            ariaLabel=\"Aggiorna valori\"\r\n            (click)=\"refreshValues.emit()\"\r\n          />\r\n\r\n          <!-- Pulisci valori -->\r\n          <ui-button\r\n            icon=\"trash-2\"\r\n            variant=\"ghost\"\r\n            size=\"xs\"\r\n            tooltip=\"Pulisci valori\"\r\n            ariaLabel=\"Pulisci valori\"\r\n            (click)=\"clearValues()\"\r\n          />\r\n\r\n          <!-- Formatta JSON -->\r\n          <ui-button\r\n            icon=\"align-left\"\r\n            variant=\"ghost\"\r\n            size=\"xs\"\r\n            tooltip=\"Formatta JSON\"\r\n            ariaLabel=\"Formatta JSON\"\r\n            (click)=\"formatJson()\"\r\n          />\r\n\r\n          <!-- Freccia toggle apertura/chiusura -->\r\n          <ui-button\r\n            [icon]=\"isOpen ? 'chevron-down' : 'chevron-up'\"\r\n            variant=\"ghost\"\r\n            size=\"xs\"\r\n            tooltip=\"Chiudi pannello\"\r\n            ariaLabel=\"Chiudi pannello\"\r\n            (click)=\"toggle.emit()\"\r\n          />\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Corpo del pannello (visibile solo quando aperto) -->\r\n      @if (isOpen) {\r\n        <div class=\"form-values-panel__body\">\r\n          <mat-form-field appearance=\"outline\" class=\"form-values-panel__textarea-wrapper\">\r\n            <mat-label>JSON</mat-label>\r\n            <textarea\r\n              matInput\r\n              class=\"form-values-panel__textarea\"\r\n              [ngModel]=\"jsonText\"\r\n              (ngModelChange)=\"onJsonInput($event)\"\r\n              rows=\"8\"\r\n              spellcheck=\"false\"\r\n              placeholder='{ \"chiave\": \"valore\" }'\r\n            ></textarea>\r\n          </mat-form-field>\r\n\r\n          <!-- Indicatore di errore di parsing -->\r\n          @if (jsonError) {\r\n            <div class=\"form-values-panel__error\">\r\n              <lucide-icon name=\"alert-triangle\" [size]=\"14\" aria-hidden=\"true\" />\r\n              <span>{{ jsonError }}</span>\r\n            </div>\r\n          }\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* Contenitore principale del pannello valori */\r\n    .form-values-panel {\r\n      border-top: 1px solid var(--ui-color-border, #e0e0e0);\r\n      background: var(--ui-color-surface, #fff);\r\n      transition: max-height 200ms ease-in-out;\r\n      overflow: hidden;\r\n    }\r\n\r\n    /* Intestazione cliccabile per toggle */\r\n    .form-values-panel__header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      padding: var(--ui-spacing-2, 8px) var(--ui-spacing-3, 12px);\r\n      cursor: pointer;\r\n      user-select: none;\r\n      transition: background-color 150ms;\r\n    }\r\n\r\n    .form-values-panel__header:hover {\r\n      background: var(--ui-color-bg-subtle, #fafafa);\r\n    }\r\n\r\n    .form-values-panel__header-left {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n    }\r\n\r\n    .form-values-panel__title {\r\n      font-weight: 600;\r\n      font-size: var(--ui-font-size-xs, 0.75rem);\r\n      color: var(--ui-color-text-secondary, #616161);\r\n      text-transform: uppercase;\r\n      letter-spacing: 0.5px;\r\n    }\r\n\r\n    .form-values-panel__header-actions {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1, 4px);\r\n    }\r\n\r\n    /* Corpo del pannello con textarea */\r\n    .form-values-panel__body {\r\n      padding: 0 var(--ui-spacing-3, 12px) var(--ui-spacing-3, 12px);\r\n    }\r\n\r\n    /* Wrapper della textarea */\r\n    .form-values-panel__textarea-wrapper {\r\n      width: 100%;\r\n    }\r\n\r\n    /* Textarea per il JSON */\r\n    .form-values-panel__textarea {\r\n      font-family: 'Fira Code', 'Consolas', 'Monaco', monospace;\r\n      font-size: var(--ui-font-size-xs, 0.75rem);\r\n      line-height: 1.5;\r\n      resize: vertical;\r\n    }\r\n\r\n    /* Messaggio di errore JSON */\r\n    .form-values-panel__error {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1, 4px);\r\n      padding: var(--ui-spacing-1, 4px) var(--ui-spacing-2, 8px);\r\n      margin-top: var(--ui-spacing-1, 4px);\r\n      border-radius: var(--ui-radius-sm, 4px);\r\n      background: color-mix(in srgb, var(--ui-color-warn, #f44336) 10%, transparent);\r\n      color: var(--ui-color-warn, #f44336);\r\n      font-size: var(--ui-font-size-xs, 0.75rem);\r\n    }\r\n  `,\r\n  ],\r\n})\r\nexport class UiEditorFormValuesPanelComponent {\r\n  /** Valori correnti del form da visualizzare. */\r\n  @Input()\r\n  set formValues(value: any) {\r\n    this._formValues = value;\r\n    this.updateJsonText();\r\n  }\r\n  get formValues(): any {\r\n    return this._formValues;\r\n  }\r\n\r\n  /** Stato di apertura/chiusura del pannello. */\r\n  @Input() isOpen = false;\r\n\r\n  /** Emesso quando i valori vengono modificati dall'utente via JSON. */\r\n  @Output() valuesChange = new EventEmitter<any>();\r\n\r\n  /** Emesso alla richiesta di aggiornamento dei valori. */\r\n  @Output() refreshValues = new EventEmitter<void>();\r\n\r\n  /** Emesso alla richiesta di apertura/chiusura del pannello. */\r\n  @Output() toggle = new EventEmitter<void>();\r\n\r\n  /** Testo JSON visualizzato nella textarea. */\r\n  jsonText = '{}';\r\n\r\n  /** Messaggio di errore per JSON non valido. */\r\n  jsonError: string | null = null;\r\n\r\n  /** @internal Valori interni del form. */\r\n  private _formValues: any = {};\r\n\r\n  /**\r\n   * Gestisce l'input dell'utente nella textarea JSON.\r\n   * Prova a parsare il JSON e aggiorna i valori se valido.\r\n   * @param text - Testo JSON inserito dall'utente\r\n   */\r\n  onJsonInput(text: string): void {\r\n    this.jsonText = text;\r\n    try {\r\n      const parsed = JSON.parse(text);\r\n      this.jsonError = null;\r\n      this.valuesChange.emit(parsed);\r\n    } catch (e) {\r\n      this.jsonError = 'JSON non valido: controlla la sintassi';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Pulisce i valori del form resettando a oggetto vuoto.\r\n   */\r\n  clearValues(): void {\r\n    this._formValues = {};\r\n    this.jsonText = '{}';\r\n    this.jsonError = null;\r\n    this.valuesChange.emit({});\r\n  }\r\n\r\n  /**\r\n   * Formatta il JSON con indentazione per una migliore leggibilita.\r\n   */\r\n  formatJson(): void {\r\n    try {\r\n      const parsed = JSON.parse(this.jsonText);\r\n      this.jsonText = JSON.stringify(parsed, null, 2);\r\n      this.jsonError = null;\r\n    } catch (e) {\r\n      this.jsonError = 'Impossibile formattare: JSON non valido';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * @internal\r\n   * Aggiorna il testo JSON dalla rappresentazione interna dei valori.\r\n   */\r\n  private updateJsonText(): void {\r\n    try {\r\n      this.jsonText = JSON.stringify(this._formValues, null, 2);\r\n      this.jsonError = null;\r\n    } catch {\r\n      this.jsonText = '{}';\r\n    }\r\n  }\r\n}\r\n"]}
|