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

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