@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,323 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/form-builder-editor
|
|
3
|
+
* Componente editor delle opzioni per campi di tipo select, radio e multiselect.
|
|
4
|
+
*
|
|
5
|
+
* Permette di aggiungere, modificare, riordinare e rimuovere le opzioni
|
|
6
|
+
* di un campo di scelta, con supporto per valore, etichetta e stato disabilitato.
|
|
7
|
+
*/
|
|
8
|
+
import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter } from '@angular/core';
|
|
9
|
+
import { FormsModule } from '@angular/forms';
|
|
10
|
+
import { MatCardModule } from '@angular/material/card';
|
|
11
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
12
|
+
import { MatInputModule } from '@angular/material/input';
|
|
13
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
14
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
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/material/checkbox";
|
|
23
|
+
import * as i6 from "@angular/material/tooltip";
|
|
24
|
+
import * as i7 from "lucide-angular";
|
|
25
|
+
/**
|
|
26
|
+
* Editor per le opzioni di un campo di scelta (select, radio, multiselect).
|
|
27
|
+
*
|
|
28
|
+
* Mostra una lista di opzioni con:
|
|
29
|
+
* - Header con titolo e pulsante per aggiungere nuove opzioni
|
|
30
|
+
* - Per ogni opzione: input valore, input etichetta, checkbox disabilitato,
|
|
31
|
+
* frecce per spostamento su/giu, pulsante elimina
|
|
32
|
+
* - Stato vuoto quando non ci sono opzioni configurate
|
|
33
|
+
*/
|
|
34
|
+
export class UiOptionsEditorComponent {
|
|
35
|
+
constructor() {
|
|
36
|
+
/**
|
|
37
|
+
* Opzioni del campo da modificare.
|
|
38
|
+
* Accetta sia un array di UiFieldOption che un valore generico
|
|
39
|
+
* (per gestire campi non ancora configurati).
|
|
40
|
+
*/
|
|
41
|
+
this.options = [];
|
|
42
|
+
/** Emette l'array aggiornato delle opzioni ad ogni modifica. */
|
|
43
|
+
this.optionsChange = new EventEmitter();
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Normalizza l'input per garantire che sia sempre un array valido.
|
|
47
|
+
* Gestisce il caso in cui le opzioni siano null, undefined o di tipo inatteso.
|
|
48
|
+
*/
|
|
49
|
+
get normalizedOptions() {
|
|
50
|
+
if (Array.isArray(this.options)) {
|
|
51
|
+
return this.options;
|
|
52
|
+
}
|
|
53
|
+
return [];
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Aggiunge una nuova opzione vuota alla lista e notifica il parent.
|
|
57
|
+
*/
|
|
58
|
+
addOption() {
|
|
59
|
+
const newOption = {
|
|
60
|
+
value: '',
|
|
61
|
+
label: '',
|
|
62
|
+
disabled: false,
|
|
63
|
+
};
|
|
64
|
+
const updated = [...this.normalizedOptions, newOption];
|
|
65
|
+
this.emitChange(updated);
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Aggiorna una proprieta specifica di un'opzione.
|
|
69
|
+
* @param index - Indice dell'opzione da aggiornare.
|
|
70
|
+
* @param field - Nome della proprieta da modificare.
|
|
71
|
+
* @param value - Nuovo valore della proprieta.
|
|
72
|
+
*/
|
|
73
|
+
updateOption(index, field, value) {
|
|
74
|
+
const updated = this.normalizedOptions.map((opt, i) => (i === index ? { ...opt, [field]: value } : opt));
|
|
75
|
+
this.emitChange(updated);
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Rimuove un'opzione dalla lista.
|
|
79
|
+
* @param index - Indice dell'opzione da rimuovere.
|
|
80
|
+
*/
|
|
81
|
+
removeOption(index) {
|
|
82
|
+
const updated = this.normalizedOptions.filter((_, i) => i !== index);
|
|
83
|
+
this.emitChange(updated);
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Sposta un'opzione in alto o in basso nella lista.
|
|
87
|
+
* @param index - Indice corrente dell'opzione.
|
|
88
|
+
* @param direction - Direzione dello spostamento: -1 (su) o +1 (giu).
|
|
89
|
+
*/
|
|
90
|
+
moveOption(index, direction) {
|
|
91
|
+
const newIndex = index + direction;
|
|
92
|
+
if (newIndex < 0 || newIndex >= this.normalizedOptions.length) {
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
const updated = [...this.normalizedOptions];
|
|
96
|
+
const [moved] = updated.splice(index, 1);
|
|
97
|
+
updated.splice(newIndex, 0, moved);
|
|
98
|
+
this.emitChange(updated);
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Emette l'array aggiornato delle opzioni.
|
|
102
|
+
* @param options - Nuovo array di opzioni.
|
|
103
|
+
*/
|
|
104
|
+
emitChange(options) {
|
|
105
|
+
this.optionsChange.emit(options);
|
|
106
|
+
}
|
|
107
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiOptionsEditorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
108
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiOptionsEditorComponent, isStandalone: true, selector: "ui-options-editor", inputs: { options: "options" }, outputs: { optionsChange: "optionsChange" }, ngImport: i0, template: `
|
|
109
|
+
<mat-card class="options-card">
|
|
110
|
+
<!-- Header -->
|
|
111
|
+
<div class="options-header">
|
|
112
|
+
<span class="options-header__title">
|
|
113
|
+
<lucide-icon name="list" [size]="16" />
|
|
114
|
+
Opzioni del Campo
|
|
115
|
+
</span>
|
|
116
|
+
<ui-button
|
|
117
|
+
icon="plus"
|
|
118
|
+
variant="ghost"
|
|
119
|
+
size="sm"
|
|
120
|
+
ariaLabel="Aggiungi opzione"
|
|
121
|
+
tooltip="Aggiungi opzione"
|
|
122
|
+
(click)="addOption()"
|
|
123
|
+
/>
|
|
124
|
+
</div>
|
|
125
|
+
|
|
126
|
+
<!-- Lista opzioni -->
|
|
127
|
+
<div class="options-list">
|
|
128
|
+
@for (option of normalizedOptions; track $index) {
|
|
129
|
+
<div class="option-row">
|
|
130
|
+
<!-- Indice opzione -->
|
|
131
|
+
<span class="option-row__index">{{ $index + 1 }}</span>
|
|
132
|
+
|
|
133
|
+
<!-- Input valore -->
|
|
134
|
+
<mat-form-field class="option-row__field" appearance="outline">
|
|
135
|
+
<mat-label>Valore</mat-label>
|
|
136
|
+
<input
|
|
137
|
+
matInput
|
|
138
|
+
[ngModel]="option.value"
|
|
139
|
+
(ngModelChange)="updateOption($index, 'value', $event)"
|
|
140
|
+
placeholder="Valore"
|
|
141
|
+
/>
|
|
142
|
+
</mat-form-field>
|
|
143
|
+
|
|
144
|
+
<!-- Input etichetta -->
|
|
145
|
+
<mat-form-field class="option-row__field" appearance="outline">
|
|
146
|
+
<mat-label>Etichetta</mat-label>
|
|
147
|
+
<input
|
|
148
|
+
matInput
|
|
149
|
+
[ngModel]="option.label"
|
|
150
|
+
(ngModelChange)="updateOption($index, 'label', $event)"
|
|
151
|
+
placeholder="Etichetta"
|
|
152
|
+
/>
|
|
153
|
+
</mat-form-field>
|
|
154
|
+
|
|
155
|
+
<!-- Checkbox disabilitato -->
|
|
156
|
+
<mat-checkbox
|
|
157
|
+
[checked]="!!option.disabled"
|
|
158
|
+
(change)="updateOption($index, 'disabled', $event.checked)"
|
|
159
|
+
matTooltip="Opzione disabilitata"
|
|
160
|
+
class="option-row__checkbox"
|
|
161
|
+
>
|
|
162
|
+
<span class="option-row__checkbox-label">Disab.</span>
|
|
163
|
+
</mat-checkbox>
|
|
164
|
+
|
|
165
|
+
<!-- Azioni di riordino e rimozione -->
|
|
166
|
+
<div class="option-row__actions">
|
|
167
|
+
<ui-button
|
|
168
|
+
icon="arrow-up"
|
|
169
|
+
variant="ghost"
|
|
170
|
+
size="xs"
|
|
171
|
+
ariaLabel="Sposta su"
|
|
172
|
+
tooltip="Sposta su"
|
|
173
|
+
[disabled]="$index === 0"
|
|
174
|
+
(click)="moveOption($index, -1)"
|
|
175
|
+
/>
|
|
176
|
+
|
|
177
|
+
<ui-button
|
|
178
|
+
icon="arrow-down"
|
|
179
|
+
variant="ghost"
|
|
180
|
+
size="xs"
|
|
181
|
+
ariaLabel="Sposta giu"
|
|
182
|
+
tooltip="Sposta giu"
|
|
183
|
+
[disabled]="$index === normalizedOptions.length - 1"
|
|
184
|
+
(click)="moveOption($index, 1)"
|
|
185
|
+
/>
|
|
186
|
+
|
|
187
|
+
<ui-button
|
|
188
|
+
icon="trash-2"
|
|
189
|
+
variant="warn"
|
|
190
|
+
size="xs"
|
|
191
|
+
ariaLabel="Elimina opzione"
|
|
192
|
+
tooltip="Elimina opzione"
|
|
193
|
+
(click)="removeOption($index)"
|
|
194
|
+
/>
|
|
195
|
+
</div>
|
|
196
|
+
</div>
|
|
197
|
+
} @empty {
|
|
198
|
+
<!-- Stato vuoto -->
|
|
199
|
+
<div class="options-empty">
|
|
200
|
+
<lucide-icon name="inbox" [size]="24" />
|
|
201
|
+
<span>Nessuna opzione configurata</span>
|
|
202
|
+
</div>
|
|
203
|
+
}
|
|
204
|
+
</div>
|
|
205
|
+
</mat-card>
|
|
206
|
+
`, isInline: true, styles: [":host{display:block}.options-card{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);overflow:hidden}.options-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-bg-subtle);border-bottom:1px solid var(--ui-color-border)}.options-header__title{display:flex;align-items:center;gap:var(--ui-spacing-2);font-size:var(--ui-font-size-sm);font-weight:600;color:var(--ui-color-text)}.options-list{padding:var(--ui-spacing-3);display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.option-row{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2);background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);transition:border-color var(--ui-transition-fast),box-shadow var(--ui-transition-fast)}.option-row:hover{border-color:var(--ui-color-primary);box-shadow:var(--ui-shadow-sm)}.option-row__index{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border-radius:50%;background:var(--ui-color-bg-subtle);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-xs);font-weight:600;flex-shrink:0}.option-row__field{flex:1;min-width:0}.option-row__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}.option-row__field ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}.option-row__checkbox{flex-shrink:0}.option-row__checkbox-label{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-secondary)}.option-row__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);flex-shrink:0}.options-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)}\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: "directive", type: i3.MatLabel, selector: "mat-label" }, { 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: MatCheckboxModule }, { kind: "component", type: i5.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i6.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i7.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 }); }
|
|
207
|
+
}
|
|
208
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiOptionsEditorComponent, decorators: [{
|
|
209
|
+
type: Component,
|
|
210
|
+
args: [{ selector: 'ui-options-editor', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [
|
|
211
|
+
FormsModule,
|
|
212
|
+
MatCardModule,
|
|
213
|
+
MatFormFieldModule,
|
|
214
|
+
MatInputModule,
|
|
215
|
+
MatCheckboxModule,
|
|
216
|
+
MatTooltipModule,
|
|
217
|
+
LucideAngularModule,
|
|
218
|
+
UiButtonComponent,
|
|
219
|
+
], template: `
|
|
220
|
+
<mat-card class="options-card">
|
|
221
|
+
<!-- Header -->
|
|
222
|
+
<div class="options-header">
|
|
223
|
+
<span class="options-header__title">
|
|
224
|
+
<lucide-icon name="list" [size]="16" />
|
|
225
|
+
Opzioni del Campo
|
|
226
|
+
</span>
|
|
227
|
+
<ui-button
|
|
228
|
+
icon="plus"
|
|
229
|
+
variant="ghost"
|
|
230
|
+
size="sm"
|
|
231
|
+
ariaLabel="Aggiungi opzione"
|
|
232
|
+
tooltip="Aggiungi opzione"
|
|
233
|
+
(click)="addOption()"
|
|
234
|
+
/>
|
|
235
|
+
</div>
|
|
236
|
+
|
|
237
|
+
<!-- Lista opzioni -->
|
|
238
|
+
<div class="options-list">
|
|
239
|
+
@for (option of normalizedOptions; track $index) {
|
|
240
|
+
<div class="option-row">
|
|
241
|
+
<!-- Indice opzione -->
|
|
242
|
+
<span class="option-row__index">{{ $index + 1 }}</span>
|
|
243
|
+
|
|
244
|
+
<!-- Input valore -->
|
|
245
|
+
<mat-form-field class="option-row__field" appearance="outline">
|
|
246
|
+
<mat-label>Valore</mat-label>
|
|
247
|
+
<input
|
|
248
|
+
matInput
|
|
249
|
+
[ngModel]="option.value"
|
|
250
|
+
(ngModelChange)="updateOption($index, 'value', $event)"
|
|
251
|
+
placeholder="Valore"
|
|
252
|
+
/>
|
|
253
|
+
</mat-form-field>
|
|
254
|
+
|
|
255
|
+
<!-- Input etichetta -->
|
|
256
|
+
<mat-form-field class="option-row__field" appearance="outline">
|
|
257
|
+
<mat-label>Etichetta</mat-label>
|
|
258
|
+
<input
|
|
259
|
+
matInput
|
|
260
|
+
[ngModel]="option.label"
|
|
261
|
+
(ngModelChange)="updateOption($index, 'label', $event)"
|
|
262
|
+
placeholder="Etichetta"
|
|
263
|
+
/>
|
|
264
|
+
</mat-form-field>
|
|
265
|
+
|
|
266
|
+
<!-- Checkbox disabilitato -->
|
|
267
|
+
<mat-checkbox
|
|
268
|
+
[checked]="!!option.disabled"
|
|
269
|
+
(change)="updateOption($index, 'disabled', $event.checked)"
|
|
270
|
+
matTooltip="Opzione disabilitata"
|
|
271
|
+
class="option-row__checkbox"
|
|
272
|
+
>
|
|
273
|
+
<span class="option-row__checkbox-label">Disab.</span>
|
|
274
|
+
</mat-checkbox>
|
|
275
|
+
|
|
276
|
+
<!-- Azioni di riordino e rimozione -->
|
|
277
|
+
<div class="option-row__actions">
|
|
278
|
+
<ui-button
|
|
279
|
+
icon="arrow-up"
|
|
280
|
+
variant="ghost"
|
|
281
|
+
size="xs"
|
|
282
|
+
ariaLabel="Sposta su"
|
|
283
|
+
tooltip="Sposta su"
|
|
284
|
+
[disabled]="$index === 0"
|
|
285
|
+
(click)="moveOption($index, -1)"
|
|
286
|
+
/>
|
|
287
|
+
|
|
288
|
+
<ui-button
|
|
289
|
+
icon="arrow-down"
|
|
290
|
+
variant="ghost"
|
|
291
|
+
size="xs"
|
|
292
|
+
ariaLabel="Sposta giu"
|
|
293
|
+
tooltip="Sposta giu"
|
|
294
|
+
[disabled]="$index === normalizedOptions.length - 1"
|
|
295
|
+
(click)="moveOption($index, 1)"
|
|
296
|
+
/>
|
|
297
|
+
|
|
298
|
+
<ui-button
|
|
299
|
+
icon="trash-2"
|
|
300
|
+
variant="warn"
|
|
301
|
+
size="xs"
|
|
302
|
+
ariaLabel="Elimina opzione"
|
|
303
|
+
tooltip="Elimina opzione"
|
|
304
|
+
(click)="removeOption($index)"
|
|
305
|
+
/>
|
|
306
|
+
</div>
|
|
307
|
+
</div>
|
|
308
|
+
} @empty {
|
|
309
|
+
<!-- Stato vuoto -->
|
|
310
|
+
<div class="options-empty">
|
|
311
|
+
<lucide-icon name="inbox" [size]="24" />
|
|
312
|
+
<span>Nessuna opzione configurata</span>
|
|
313
|
+
</div>
|
|
314
|
+
}
|
|
315
|
+
</div>
|
|
316
|
+
</mat-card>
|
|
317
|
+
`, styles: [":host{display:block}.options-card{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);overflow:hidden}.options-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-bg-subtle);border-bottom:1px solid var(--ui-color-border)}.options-header__title{display:flex;align-items:center;gap:var(--ui-spacing-2);font-size:var(--ui-font-size-sm);font-weight:600;color:var(--ui-color-text)}.options-list{padding:var(--ui-spacing-3);display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.option-row{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2);background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);transition:border-color var(--ui-transition-fast),box-shadow var(--ui-transition-fast)}.option-row:hover{border-color:var(--ui-color-primary);box-shadow:var(--ui-shadow-sm)}.option-row__index{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border-radius:50%;background:var(--ui-color-bg-subtle);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-xs);font-weight:600;flex-shrink:0}.option-row__field{flex:1;min-width:0}.option-row__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}.option-row__field ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}.option-row__checkbox{flex-shrink:0}.option-row__checkbox-label{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-secondary)}.option-row__actions{display:flex;align-items:center;gap:var(--ui-spacing-1);flex-shrink:0}.options-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)}\n"] }]
|
|
318
|
+
}], propDecorators: { options: [{
|
|
319
|
+
type: Input
|
|
320
|
+
}], optionsChange: [{
|
|
321
|
+
type: Output
|
|
322
|
+
}] } });
|
|
323
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"options-editor.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/form-builder-editor\r\n * Componente editor delle opzioni per campi di tipo select, radio e multiselect.\r\n *\r\n * Permette di aggiungere, modificare, riordinare e rimuovere le opzioni\r\n * di un campo di scelta, con supporto per valore, etichetta e stato disabilitato.\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 { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\n\r\nimport { UiButtonComponent } from '../../../button/index';\r\nimport { UiFieldOption } from '../../../form-builder/types/index';\r\n\r\n/**\r\n * Editor per le opzioni di un campo di scelta (select, radio, multiselect).\r\n *\r\n * Mostra una lista di opzioni con:\r\n * - Header con titolo e pulsante per aggiungere nuove opzioni\r\n * - Per ogni opzione: input valore, input etichetta, checkbox disabilitato,\r\n *   frecce per spostamento su/giu, pulsante elimina\r\n * - Stato vuoto quando non ci sono opzioni configurate\r\n */\r\n@Component({\r\n  selector: 'ui-options-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    MatCheckboxModule,\r\n    MatTooltipModule,\r\n    LucideAngularModule,\r\n    UiButtonComponent,\r\n  ],\r\n  template: `\r\n    <mat-card class=\"options-card\">\r\n      <!-- Header -->\r\n      <div class=\"options-header\">\r\n        <span class=\"options-header__title\">\r\n          <lucide-icon name=\"list\" [size]=\"16\" />\r\n          Opzioni del Campo\r\n        </span>\r\n        <ui-button\r\n          icon=\"plus\"\r\n          variant=\"ghost\"\r\n          size=\"sm\"\r\n          ariaLabel=\"Aggiungi opzione\"\r\n          tooltip=\"Aggiungi opzione\"\r\n          (click)=\"addOption()\"\r\n        />\r\n      </div>\r\n\r\n      <!-- Lista opzioni -->\r\n      <div class=\"options-list\">\r\n        @for (option of normalizedOptions; track $index) {\r\n          <div class=\"option-row\">\r\n            <!-- Indice opzione -->\r\n            <span class=\"option-row__index\">{{ $index + 1 }}</span>\r\n\r\n            <!-- Input valore -->\r\n            <mat-form-field class=\"option-row__field\" appearance=\"outline\">\r\n              <mat-label>Valore</mat-label>\r\n              <input\r\n                matInput\r\n                [ngModel]=\"option.value\"\r\n                (ngModelChange)=\"updateOption($index, 'value', $event)\"\r\n                placeholder=\"Valore\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- Input etichetta -->\r\n            <mat-form-field class=\"option-row__field\" appearance=\"outline\">\r\n              <mat-label>Etichetta</mat-label>\r\n              <input\r\n                matInput\r\n                [ngModel]=\"option.label\"\r\n                (ngModelChange)=\"updateOption($index, 'label', $event)\"\r\n                placeholder=\"Etichetta\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- Checkbox disabilitato -->\r\n            <mat-checkbox\r\n              [checked]=\"!!option.disabled\"\r\n              (change)=\"updateOption($index, 'disabled', $event.checked)\"\r\n              matTooltip=\"Opzione disabilitata\"\r\n              class=\"option-row__checkbox\"\r\n            >\r\n              <span class=\"option-row__checkbox-label\">Disab.</span>\r\n            </mat-checkbox>\r\n\r\n            <!-- Azioni di riordino e rimozione -->\r\n            <div class=\"option-row__actions\">\r\n              <ui-button\r\n                icon=\"arrow-up\"\r\n                variant=\"ghost\"\r\n                size=\"xs\"\r\n                ariaLabel=\"Sposta su\"\r\n                tooltip=\"Sposta su\"\r\n                [disabled]=\"$index === 0\"\r\n                (click)=\"moveOption($index, -1)\"\r\n              />\r\n\r\n              <ui-button\r\n                icon=\"arrow-down\"\r\n                variant=\"ghost\"\r\n                size=\"xs\"\r\n                ariaLabel=\"Sposta giu\"\r\n                tooltip=\"Sposta giu\"\r\n                [disabled]=\"$index === normalizedOptions.length - 1\"\r\n                (click)=\"moveOption($index, 1)\"\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 opzione\"\r\n                tooltip=\"Elimina opzione\"\r\n                (click)=\"removeOption($index)\"\r\n              />\r\n            </div>\r\n          </div>\r\n        } @empty {\r\n          <!-- Stato vuoto -->\r\n          <div class=\"options-empty\">\r\n            <lucide-icon name=\"inbox\" [size]=\"24\" />\r\n            <span>Nessuna opzione configurata</span>\r\n          </div>\r\n        }\r\n      </div>\r\n    </mat-card>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* ─── Card opzioni ──────────────────────────────────────── */\r\n\r\n    :host {\r\n      display: block;\r\n    }\r\n\r\n    .options-card {\r\n      border: 1px solid var(--ui-color-border);\r\n      border-radius: var(--ui-radius-md);\r\n      overflow: hidden;\r\n    }\r\n\r\n    /* ─── Header ────────────────────────────────────────────── */\r\n\r\n    .options-header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      padding: var(--ui-spacing-3) var(--ui-spacing-4);\r\n      background: var(--ui-color-bg-subtle);\r\n      border-bottom: 1px solid var(--ui-color-border);\r\n    }\r\n\r\n    .options-header__title {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n      font-size: var(--ui-font-size-sm);\r\n      font-weight: 600;\r\n      color: var(--ui-color-text);\r\n    }\r\n\r\n    /* ─── Lista opzioni ─────────────────────────────────────── */\r\n\r\n    .options-list {\r\n      padding: var(--ui-spacing-3);\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: var(--ui-spacing-2);\r\n    }\r\n\r\n    /* ─── Riga opzione ──────────────────────────────────────── */\r\n\r\n    .option-row {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n      padding: var(--ui-spacing-2);\r\n      background: var(--ui-color-surface);\r\n      border: 1px solid var(--ui-color-border);\r\n      border-radius: var(--ui-radius-sm);\r\n      transition: border-color var(--ui-transition-fast),\r\n                  box-shadow var(--ui-transition-fast);\r\n    }\r\n\r\n    .option-row:hover {\r\n      border-color: var(--ui-color-primary);\r\n      box-shadow: var(--ui-shadow-sm);\r\n    }\r\n\r\n    .option-row__index {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      width: 24px;\r\n      height: 24px;\r\n      border-radius: 50%;\r\n      background: var(--ui-color-bg-subtle);\r\n      color: var(--ui-color-text-muted);\r\n      font-size: var(--ui-font-size-xs);\r\n      font-weight: 600;\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    .option-row__field {\r\n      flex: 1;\r\n      min-width: 0;\r\n    }\r\n\r\n    /* Riduce l'altezza dei form field inline */\r\n    .option-row__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    .option-row__field ::ng-deep .mat-mdc-form-field-subscript-wrapper {\r\n      display: none;\r\n    }\r\n\r\n    .option-row__checkbox {\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    .option-row__checkbox-label {\r\n      font-size: var(--ui-font-size-xs);\r\n      color: var(--ui-color-text-secondary);\r\n    }\r\n\r\n    .option-row__actions {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1);\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    /* ─── Stato vuoto ───────────────────────────────────────── */\r\n\r\n    .options-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  `,\r\n  ],\r\n})\r\nexport class UiOptionsEditorComponent {\r\n  /**\r\n   * Opzioni del campo da modificare.\r\n   * Accetta sia un array di UiFieldOption che un valore generico\r\n   * (per gestire campi non ancora configurati).\r\n   */\r\n  @Input() options: UiFieldOption[] | any = [];\r\n\r\n  /** Emette l'array aggiornato delle opzioni ad ogni modifica. */\r\n  @Output() optionsChange = new EventEmitter<UiFieldOption[]>();\r\n\r\n  /**\r\n   * Normalizza l'input per garantire che sia sempre un array valido.\r\n   * Gestisce il caso in cui le opzioni siano null, undefined o di tipo inatteso.\r\n   */\r\n  get normalizedOptions(): UiFieldOption[] {\r\n    if (Array.isArray(this.options)) {\r\n      return this.options;\r\n    }\r\n    return [];\r\n  }\r\n\r\n  /**\r\n   * Aggiunge una nuova opzione vuota alla lista e notifica il parent.\r\n   */\r\n  addOption(): void {\r\n    const newOption: UiFieldOption = {\r\n      value: '',\r\n      label: '',\r\n      disabled: false,\r\n    };\r\n    const updated = [...this.normalizedOptions, newOption];\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Aggiorna una proprieta specifica di un'opzione.\r\n   * @param index - Indice dell'opzione da aggiornare.\r\n   * @param field - Nome della proprieta da modificare.\r\n   * @param value - Nuovo valore della proprieta.\r\n   */\r\n  updateOption(index: number, field: keyof UiFieldOption, value: any): void {\r\n    const updated = this.normalizedOptions.map((opt, i) => (i === index ? { ...opt, [field]: value } : opt));\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Rimuove un'opzione dalla lista.\r\n   * @param index - Indice dell'opzione da rimuovere.\r\n   */\r\n  removeOption(index: number): void {\r\n    const updated = this.normalizedOptions.filter((_, i) => i !== index);\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Sposta un'opzione in alto o in basso nella lista.\r\n   * @param index - Indice corrente dell'opzione.\r\n   * @param direction - Direzione dello spostamento: -1 (su) o +1 (giu).\r\n   */\r\n  moveOption(index: number, direction: -1 | 1): void {\r\n    const newIndex = index + direction;\r\n    if (newIndex < 0 || newIndex >= this.normalizedOptions.length) {\r\n      return;\r\n    }\r\n    const updated = [...this.normalizedOptions];\r\n    const [moved] = updated.splice(index, 1);\r\n    updated.splice(newIndex, 0, moved);\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Emette l'array aggiornato delle opzioni.\r\n   * @param options - Nuovo array di opzioni.\r\n   */\r\n  private emitChange(options: UiFieldOption[]): void {\r\n    this.optionsChange.emit(options);\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy, ChangeDetectorRef, Input, Output, EventEmitter, ViewChild, inject, } from '@angular/core';
|
|
2
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
3
|
+
import { UiButtonComponent } from '../../../button/index';
|
|
4
|
+
import { UiFormBuilderComponent } from '../../../form-builder/form-builder.component';
|
|
5
|
+
import { UiEditorFormValuesPanelComponent } from '../form-values-panel/form-values-panel.component';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "lucide-angular";
|
|
8
|
+
/**
|
|
9
|
+
* Contenitore per l'anteprima live del form in fase di editing.
|
|
10
|
+
*
|
|
11
|
+
* Mostra una preview del form builder generato dallo schema corrente,
|
|
12
|
+
* con possibilita di aggiornamento forzato e pannello dei valori.
|
|
13
|
+
*
|
|
14
|
+
* @selector ui-editor-preview-container
|
|
15
|
+
*/
|
|
16
|
+
export class UiEditorPreviewContainerComponent {
|
|
17
|
+
constructor() {
|
|
18
|
+
/** @internal Change detector per OnPush. */
|
|
19
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
20
|
+
/** Schema del form ricevuto dal parent (cambia ad ogni modifica nell'editor). */
|
|
21
|
+
this.schema = null;
|
|
22
|
+
/**
|
|
23
|
+
* Chiave per forzare la ri-creazione del form builder.
|
|
24
|
+
* Viene incrementata dal componente padre per invalidare l'anteprima.
|
|
25
|
+
*/
|
|
26
|
+
this.key = 0;
|
|
27
|
+
/** Emesso alla richiesta di aggiornamento forzato dell'anteprima. */
|
|
28
|
+
this.refresh = new EventEmitter();
|
|
29
|
+
/**
|
|
30
|
+
* Snapshot immutabile dello schema usato per il rendering.
|
|
31
|
+
* Viene aggiornato SOLO al primo caricamento, al cambio tab
|
|
32
|
+
* (quando `key` cambia) o al refresh esplicito.
|
|
33
|
+
* Questo evita che le modifiche nell'editor ricostruiscano
|
|
34
|
+
* continuamente il form, perdendo lo stato dell'utente.
|
|
35
|
+
*/
|
|
36
|
+
this.previewSchema = null;
|
|
37
|
+
/** Stato di apertura del pannello valori in basso. */
|
|
38
|
+
this.isValuesPanelOpen = false;
|
|
39
|
+
/** Valori correnti del form nell'anteprima. */
|
|
40
|
+
this.currentFormValues = {};
|
|
41
|
+
/** @internal Ultimo valore di `key` applicato allo snapshot. */
|
|
42
|
+
this.lastAppliedKey = -1;
|
|
43
|
+
}
|
|
44
|
+
ngOnChanges(changes) {
|
|
45
|
+
const keyChanged = changes['key'] && this.key !== this.lastAppliedKey;
|
|
46
|
+
const firstSchema = changes['schema']?.firstChange;
|
|
47
|
+
if (keyChanged || firstSchema) {
|
|
48
|
+
this.applySchemaSnapshot();
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Riceve i valori aggiornati dal form builder ad ogni interazione.
|
|
53
|
+
* @param values - Valori correnti del form (inclusi campi disabilitati)
|
|
54
|
+
*/
|
|
55
|
+
onFormBuilderValueChange(values) {
|
|
56
|
+
this.currentFormValues = values;
|
|
57
|
+
this.cdr.markForCheck();
|
|
58
|
+
}
|
|
59
|
+
/** Alterna la visibilita del pannello dei valori del form. */
|
|
60
|
+
toggleValuesPanel() {
|
|
61
|
+
this.isValuesPanelOpen = !this.isValuesPanelOpen;
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Gestisce il cambiamento dei valori dal pannello JSON.
|
|
65
|
+
* @param values - Nuovi valori parsati dal JSON
|
|
66
|
+
*/
|
|
67
|
+
onFormValuesChange(values) {
|
|
68
|
+
this.currentFormValues = values;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Aggiorna i valori leggendo lo stato corrente dal form builder.
|
|
72
|
+
*/
|
|
73
|
+
onRefreshValues() {
|
|
74
|
+
if (this.formBuilderRef) {
|
|
75
|
+
this.currentFormValues = this.formBuilderRef.getFormValue();
|
|
76
|
+
this.cdr.markForCheck();
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* @internal
|
|
81
|
+
* Crea una copia profonda dello schema corrente come snapshot
|
|
82
|
+
* per il rendering dell'anteprima. L'id viene reso univoco
|
|
83
|
+
* per forzare il rebuild del form builder (che ricostruisce
|
|
84
|
+
* solo quando `schema.id` cambia).
|
|
85
|
+
*/
|
|
86
|
+
applySchemaSnapshot() {
|
|
87
|
+
if (this.schema) {
|
|
88
|
+
const snapshot = structuredClone(this.schema);
|
|
89
|
+
snapshot.id = `${snapshot.id}__preview_${this.key}`;
|
|
90
|
+
this.previewSchema = snapshot;
|
|
91
|
+
}
|
|
92
|
+
else {
|
|
93
|
+
this.previewSchema = null;
|
|
94
|
+
}
|
|
95
|
+
this.lastAppliedKey = this.key;
|
|
96
|
+
this.currentFormValues = {};
|
|
97
|
+
}
|
|
98
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorPreviewContainerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
99
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiEditorPreviewContainerComponent, isStandalone: true, selector: "ui-editor-preview-container", inputs: { schema: "schema", key: "key" }, outputs: { refresh: "refresh" }, viewQueries: [{ propertyName: "formBuilderRef", first: true, predicate: UiFormBuilderComponent, descendants: true }], usesOnChanges: true, ngImport: i0, template: `
|
|
100
|
+
<div class="preview-container">
|
|
101
|
+
<!-- Intestazione con azioni -->
|
|
102
|
+
<div class="preview-container__header">
|
|
103
|
+
<div class="preview-container__header-left">
|
|
104
|
+
<lucide-icon name="eye" [size]="18" aria-hidden="true" />
|
|
105
|
+
<span class="preview-container__title">Anteprima Form</span>
|
|
106
|
+
</div>
|
|
107
|
+
|
|
108
|
+
<div class="preview-container__header-actions">
|
|
109
|
+
<!-- Aggiorna anteprima -->
|
|
110
|
+
<ui-button
|
|
111
|
+
icon="refresh-cw"
|
|
112
|
+
variant="ghost"
|
|
113
|
+
size="sm"
|
|
114
|
+
tooltip="Aggiorna anteprima"
|
|
115
|
+
ariaLabel="Aggiorna anteprima"
|
|
116
|
+
(click)="refresh.emit()"
|
|
117
|
+
/>
|
|
118
|
+
|
|
119
|
+
<!-- Mostra/nascondi pannello valori -->
|
|
120
|
+
<ui-button
|
|
121
|
+
icon="code"
|
|
122
|
+
variant="ghost"
|
|
123
|
+
size="sm"
|
|
124
|
+
tooltip="Mostra/nascondi valori form"
|
|
125
|
+
ariaLabel="Mostra/nascondi valori form"
|
|
126
|
+
(click)="toggleValuesPanel()"
|
|
127
|
+
/>
|
|
128
|
+
</div>
|
|
129
|
+
</div>
|
|
130
|
+
|
|
131
|
+
<!-- Contenuto anteprima -->
|
|
132
|
+
<div class="preview-container__content">
|
|
133
|
+
@if (!previewSchema || !previewSchema.sections || previewSchema.sections.length === 0) {
|
|
134
|
+
<!-- Stato vuoto: nessuno schema -->
|
|
135
|
+
<div class="preview-container__empty">
|
|
136
|
+
<lucide-icon name="layout-template" [size]="40" aria-hidden="true" />
|
|
137
|
+
<p class="preview-container__empty-text">
|
|
138
|
+
Aggiungi sezioni e campi per visualizzare l'anteprima
|
|
139
|
+
</p>
|
|
140
|
+
</div>
|
|
141
|
+
} @else {
|
|
142
|
+
<!-- Rendering del form builder con snapshot dello schema -->
|
|
143
|
+
<div class="preview-container__form-wrapper">
|
|
144
|
+
<ui-form-builder
|
|
145
|
+
[schema]="previewSchema"
|
|
146
|
+
(valueChange)="onFormBuilderValueChange($event)"
|
|
147
|
+
/>
|
|
148
|
+
</div>
|
|
149
|
+
}
|
|
150
|
+
</div>
|
|
151
|
+
|
|
152
|
+
<!-- Pannello valori in basso -->
|
|
153
|
+
<ui-editor-form-values-panel
|
|
154
|
+
[formValues]="currentFormValues"
|
|
155
|
+
[isOpen]="isValuesPanelOpen"
|
|
156
|
+
(valuesChange)="onFormValuesChange($event)"
|
|
157
|
+
(refreshValues)="onRefreshValues()"
|
|
158
|
+
(toggle)="toggleValuesPanel()"
|
|
159
|
+
/>
|
|
160
|
+
</div>
|
|
161
|
+
`, isInline: true, styles: [".preview-container{display:flex;flex-direction:column;height:100%;background:var(--ui-color-bg-subtle, #fafafa);position:relative}.preview-container__header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2, 8px) var(--ui-spacing-4, 16px);border-bottom:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-surface, #fff)}.preview-container__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.preview-container__title{font-weight:600;font-size:var(--ui-font-size-sm, .875rem);color:var(--ui-color-text, #212121)}.preview-container__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px)}.preview-container__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-4, 16px)}.preview-container__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-3, 12px);padding:var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);text-align:center;color:var(--ui-color-text-muted, #9e9e9e)}.preview-container__empty-text{font-size:var(--ui-font-size-sm, .875rem);margin:0}.preview-container__form-wrapper{background:var(--ui-color-surface, #fff);border-radius:var(--ui-radius-md, 8px);padding:var(--ui-spacing-4, 16px);box-shadow:var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,.1))}\n"], dependencies: [{ kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.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"] }, { kind: "component", type: UiFormBuilderComponent, selector: "ui-form-builder", inputs: ["schema", "initialData", "readonly", "disabled", "buttonsOverride", "loadingFor"], outputs: ["valueChange", "validationChange", "formSubmit", "formReset", "customEvent"] }, { kind: "component", type: UiEditorFormValuesPanelComponent, selector: "ui-editor-form-values-panel", inputs: ["formValues", "isOpen"], outputs: ["valuesChange", "refreshValues", "toggle"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
162
|
+
}
|
|
163
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorPreviewContainerComponent, decorators: [{
|
|
164
|
+
type: Component,
|
|
165
|
+
args: [{ selector: 'ui-editor-preview-container', standalone: true, imports: [LucideAngularModule, UiButtonComponent, UiFormBuilderComponent, UiEditorFormValuesPanelComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
166
|
+
<div class="preview-container">
|
|
167
|
+
<!-- Intestazione con azioni -->
|
|
168
|
+
<div class="preview-container__header">
|
|
169
|
+
<div class="preview-container__header-left">
|
|
170
|
+
<lucide-icon name="eye" [size]="18" aria-hidden="true" />
|
|
171
|
+
<span class="preview-container__title">Anteprima Form</span>
|
|
172
|
+
</div>
|
|
173
|
+
|
|
174
|
+
<div class="preview-container__header-actions">
|
|
175
|
+
<!-- Aggiorna anteprima -->
|
|
176
|
+
<ui-button
|
|
177
|
+
icon="refresh-cw"
|
|
178
|
+
variant="ghost"
|
|
179
|
+
size="sm"
|
|
180
|
+
tooltip="Aggiorna anteprima"
|
|
181
|
+
ariaLabel="Aggiorna anteprima"
|
|
182
|
+
(click)="refresh.emit()"
|
|
183
|
+
/>
|
|
184
|
+
|
|
185
|
+
<!-- Mostra/nascondi pannello valori -->
|
|
186
|
+
<ui-button
|
|
187
|
+
icon="code"
|
|
188
|
+
variant="ghost"
|
|
189
|
+
size="sm"
|
|
190
|
+
tooltip="Mostra/nascondi valori form"
|
|
191
|
+
ariaLabel="Mostra/nascondi valori form"
|
|
192
|
+
(click)="toggleValuesPanel()"
|
|
193
|
+
/>
|
|
194
|
+
</div>
|
|
195
|
+
</div>
|
|
196
|
+
|
|
197
|
+
<!-- Contenuto anteprima -->
|
|
198
|
+
<div class="preview-container__content">
|
|
199
|
+
@if (!previewSchema || !previewSchema.sections || previewSchema.sections.length === 0) {
|
|
200
|
+
<!-- Stato vuoto: nessuno schema -->
|
|
201
|
+
<div class="preview-container__empty">
|
|
202
|
+
<lucide-icon name="layout-template" [size]="40" aria-hidden="true" />
|
|
203
|
+
<p class="preview-container__empty-text">
|
|
204
|
+
Aggiungi sezioni e campi per visualizzare l'anteprima
|
|
205
|
+
</p>
|
|
206
|
+
</div>
|
|
207
|
+
} @else {
|
|
208
|
+
<!-- Rendering del form builder con snapshot dello schema -->
|
|
209
|
+
<div class="preview-container__form-wrapper">
|
|
210
|
+
<ui-form-builder
|
|
211
|
+
[schema]="previewSchema"
|
|
212
|
+
(valueChange)="onFormBuilderValueChange($event)"
|
|
213
|
+
/>
|
|
214
|
+
</div>
|
|
215
|
+
}
|
|
216
|
+
</div>
|
|
217
|
+
|
|
218
|
+
<!-- Pannello valori in basso -->
|
|
219
|
+
<ui-editor-form-values-panel
|
|
220
|
+
[formValues]="currentFormValues"
|
|
221
|
+
[isOpen]="isValuesPanelOpen"
|
|
222
|
+
(valuesChange)="onFormValuesChange($event)"
|
|
223
|
+
(refreshValues)="onRefreshValues()"
|
|
224
|
+
(toggle)="toggleValuesPanel()"
|
|
225
|
+
/>
|
|
226
|
+
</div>
|
|
227
|
+
`, styles: [".preview-container{display:flex;flex-direction:column;height:100%;background:var(--ui-color-bg-subtle, #fafafa);position:relative}.preview-container__header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2, 8px) var(--ui-spacing-4, 16px);border-bottom:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-surface, #fff)}.preview-container__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px)}.preview-container__title{font-weight:600;font-size:var(--ui-font-size-sm, .875rem);color:var(--ui-color-text, #212121)}.preview-container__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1, 4px)}.preview-container__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-4, 16px)}.preview-container__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-3, 12px);padding:var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);text-align:center;color:var(--ui-color-text-muted, #9e9e9e)}.preview-container__empty-text{font-size:var(--ui-font-size-sm, .875rem);margin:0}.preview-container__form-wrapper{background:var(--ui-color-surface, #fff);border-radius:var(--ui-radius-md, 8px);padding:var(--ui-spacing-4, 16px);box-shadow:var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,.1))}\n"] }]
|
|
228
|
+
}], propDecorators: { formBuilderRef: [{
|
|
229
|
+
type: ViewChild,
|
|
230
|
+
args: [UiFormBuilderComponent]
|
|
231
|
+
}], schema: [{
|
|
232
|
+
type: Input
|
|
233
|
+
}], key: [{
|
|
234
|
+
type: Input
|
|
235
|
+
}], refresh: [{
|
|
236
|
+
type: Output
|
|
237
|
+
}] } });
|
|
238
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preview-container.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  OnChanges,\r\n  SimpleChanges,\r\n  ViewChild,\r\n  inject,\r\n} from '@angular/core';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiButtonComponent } from '../../../button/index';\r\n\r\nimport { UiFormSchema } from '../../../form-builder/types/index';\r\nimport { UiFormBuilderComponent } from '../../../form-builder/form-builder.component';\r\nimport { UiEditorFormValuesPanelComponent } from '../form-values-panel/form-values-panel.component';\r\n\r\n/**\r\n * Contenitore per l'anteprima live del form in fase di editing.\r\n *\r\n * Mostra una preview del form builder generato dallo schema corrente,\r\n * con possibilita di aggiornamento forzato e pannello dei valori.\r\n *\r\n * @selector ui-editor-preview-container\r\n */\r\n@Component({\r\n  selector: 'ui-editor-preview-container',\r\n  standalone: true,\r\n  imports: [LucideAngularModule, UiButtonComponent, UiFormBuilderComponent, UiEditorFormValuesPanelComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"preview-container\">\r\n      <!-- Intestazione con azioni -->\r\n      <div class=\"preview-container__header\">\r\n        <div class=\"preview-container__header-left\">\r\n          <lucide-icon name=\"eye\" [size]=\"18\" aria-hidden=\"true\" />\r\n          <span class=\"preview-container__title\">Anteprima Form</span>\r\n        </div>\r\n\r\n        <div class=\"preview-container__header-actions\">\r\n          <!-- Aggiorna anteprima -->\r\n          <ui-button\r\n            icon=\"refresh-cw\"\r\n            variant=\"ghost\"\r\n            size=\"sm\"\r\n            tooltip=\"Aggiorna anteprima\"\r\n            ariaLabel=\"Aggiorna anteprima\"\r\n            (click)=\"refresh.emit()\"\r\n          />\r\n\r\n          <!-- Mostra/nascondi pannello valori -->\r\n          <ui-button\r\n            icon=\"code\"\r\n            variant=\"ghost\"\r\n            size=\"sm\"\r\n            tooltip=\"Mostra/nascondi valori form\"\r\n            ariaLabel=\"Mostra/nascondi valori form\"\r\n            (click)=\"toggleValuesPanel()\"\r\n          />\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Contenuto anteprima -->\r\n      <div class=\"preview-container__content\">\r\n        @if (!previewSchema || !previewSchema.sections || previewSchema.sections.length === 0) {\r\n          <!-- Stato vuoto: nessuno schema -->\r\n          <div class=\"preview-container__empty\">\r\n            <lucide-icon name=\"layout-template\" [size]=\"40\" aria-hidden=\"true\" />\r\n            <p class=\"preview-container__empty-text\">\r\n              Aggiungi sezioni e campi per visualizzare l'anteprima\r\n            </p>\r\n          </div>\r\n        } @else {\r\n          <!-- Rendering del form builder con snapshot dello schema -->\r\n          <div class=\"preview-container__form-wrapper\">\r\n            <ui-form-builder\r\n              [schema]=\"previewSchema\"\r\n              (valueChange)=\"onFormBuilderValueChange($event)\"\r\n            />\r\n          </div>\r\n        }\r\n      </div>\r\n\r\n      <!-- Pannello valori in basso -->\r\n      <ui-editor-form-values-panel\r\n        [formValues]=\"currentFormValues\"\r\n        [isOpen]=\"isValuesPanelOpen\"\r\n        (valuesChange)=\"onFormValuesChange($event)\"\r\n        (refreshValues)=\"onRefreshValues()\"\r\n        (toggle)=\"toggleValuesPanel()\"\r\n      />\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* Contenitore principale dell'anteprima */\r\n    .preview-container {\r\n      display: flex;\r\n      flex-direction: column;\r\n      height: 100%;\r\n      background: var(--ui-color-bg-subtle, #fafafa);\r\n      position: relative;\r\n    }\r\n\r\n    /* Intestazione dell'anteprima */\r\n    .preview-container__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-4, 16px);\r\n      border-bottom: 1px solid var(--ui-color-border, #e0e0e0);\r\n      background: var(--ui-color-surface, #fff);\r\n    }\r\n\r\n    .preview-container__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    .preview-container__title {\r\n      font-weight: 600;\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      color: var(--ui-color-text, #212121);\r\n    }\r\n\r\n    .preview-container__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    /* Area di contenuto scrollabile */\r\n    .preview-container__content {\r\n      flex: 1;\r\n      overflow-y: auto;\r\n      padding: var(--ui-spacing-4, 16px);\r\n    }\r\n\r\n    /* Stato vuoto: nessuno schema disponibile */\r\n    .preview-container__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-3, 12px);\r\n      padding: var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);\r\n      text-align: center;\r\n      color: var(--ui-color-text-muted, #9e9e9e);\r\n    }\r\n\r\n    .preview-container__empty-text {\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      margin: 0;\r\n    }\r\n\r\n    /* Wrapper del form builder renderizzato */\r\n    .preview-container__form-wrapper {\r\n      background: var(--ui-color-surface, #fff);\r\n      border-radius: var(--ui-radius-md, 8px);\r\n      padding: var(--ui-spacing-4, 16px);\r\n      box-shadow: var(--ui-shadow-sm, 0 1px 3px rgba(0,0,0,0.1));\r\n    }\r\n  `,\r\n  ],\r\n})\r\nexport class UiEditorPreviewContainerComponent implements OnChanges {\r\n  /** @internal Change detector per OnPush. */\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n\r\n  /** Riferimento all'istanza del form builder per leggere i valori. */\r\n  @ViewChild(UiFormBuilderComponent) formBuilderRef?: UiFormBuilderComponent;\r\n\r\n  /** Schema del form ricevuto dal parent (cambia ad ogni modifica nell'editor). */\r\n  @Input() schema: UiFormSchema | null = null;\r\n\r\n  /**\r\n   * Chiave per forzare la ri-creazione del form builder.\r\n   * Viene incrementata dal componente padre per invalidare l'anteprima.\r\n   */\r\n  @Input() key: number = 0;\r\n\r\n  /** Emesso alla richiesta di aggiornamento forzato dell'anteprima. */\r\n  @Output() refresh = new EventEmitter<void>();\r\n\r\n  /**\r\n   * Snapshot immutabile dello schema usato per il rendering.\r\n   * Viene aggiornato SOLO al primo caricamento, al cambio tab\r\n   * (quando `key` cambia) o al refresh esplicito.\r\n   * Questo evita che le modifiche nell'editor ricostruiscano\r\n   * continuamente il form, perdendo lo stato dell'utente.\r\n   */\r\n  previewSchema: UiFormSchema | null = null;\r\n\r\n  /** Stato di apertura del pannello valori in basso. */\r\n  isValuesPanelOpen = false;\r\n\r\n  /** Valori correnti del form nell'anteprima. */\r\n  currentFormValues: any = {};\r\n\r\n  /** @internal Ultimo valore di `key` applicato allo snapshot. */\r\n  private lastAppliedKey = -1;\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    const keyChanged = changes['key'] && this.key !== this.lastAppliedKey;\r\n    const firstSchema = changes['schema']?.firstChange;\r\n\r\n    if (keyChanged || firstSchema) {\r\n      this.applySchemaSnapshot();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Riceve i valori aggiornati dal form builder ad ogni interazione.\r\n   * @param values - Valori correnti del form (inclusi campi disabilitati)\r\n   */\r\n  onFormBuilderValueChange(values: any): void {\r\n    this.currentFormValues = values;\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  /** Alterna la visibilita del pannello dei valori del form. */\r\n  toggleValuesPanel(): void {\r\n    this.isValuesPanelOpen = !this.isValuesPanelOpen;\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambiamento dei valori dal pannello JSON.\r\n   * @param values - Nuovi valori parsati dal JSON\r\n   */\r\n  onFormValuesChange(values: any): void {\r\n    this.currentFormValues = values;\r\n  }\r\n\r\n  /**\r\n   * Aggiorna i valori leggendo lo stato corrente dal form builder.\r\n   */\r\n  onRefreshValues(): void {\r\n    if (this.formBuilderRef) {\r\n      this.currentFormValues = this.formBuilderRef.getFormValue();\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * @internal\r\n   * Crea una copia profonda dello schema corrente come snapshot\r\n   * per il rendering dell'anteprima. L'id viene reso univoco\r\n   * per forzare il rebuild del form builder (che ricostruisce\r\n   * solo quando `schema.id` cambia).\r\n   */\r\n  private applySchemaSnapshot(): void {\r\n    if (this.schema) {\r\n      const snapshot = structuredClone(this.schema);\r\n      snapshot.id = `${snapshot.id}__preview_${this.key}`;\r\n      this.previewSchema = snapshot;\r\n    } else {\r\n      this.previewSchema = null;\r\n    }\r\n    this.lastAppliedKey = this.key;\r\n    this.currentFormValues = {};\r\n  }\r\n}\r\n"]}
|