@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,667 @@
1
+ import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy } from '@angular/core';
2
+ import { FormsModule } from '@angular/forms';
3
+ import { MatCardModule } from '@angular/material/card';
4
+ import { MatFormFieldModule } from '@angular/material/form-field';
5
+ import { MatInputModule } from '@angular/material/input';
6
+ import { MatSelectModule } from '@angular/material/select';
7
+ import { LucideAngularModule } from 'lucide-angular';
8
+ import { UiButtonComponent } from '../../../button/index';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "@angular/forms";
11
+ import * as i2 from "@angular/material/card";
12
+ import * as i3 from "@angular/material/form-field";
13
+ import * as i4 from "@angular/material/input";
14
+ import * as i5 from "@angular/material/select";
15
+ import * as i6 from "@angular/material/core";
16
+ import * as i7 from "lucide-angular";
17
+ /**
18
+ * Editor per le condizioni di visibilita/disabilitazione dei campi.
19
+ *
20
+ * Permette di configurare condizioni con editor visuale o JSON.
21
+ * Supporta operatori di confronto multipli, logica AND/OR
22
+ * e selezione dei valori dalle opzioni dei campi disponibili.
23
+ *
24
+ * @selector ui-condition-editor
25
+ */
26
+ export class UiConditionEditorComponent {
27
+ constructor() {
28
+ /** Lista delle condizioni da editare. */
29
+ this.conditions = [];
30
+ /** Titolo visualizzato nell'intestazione. */
31
+ this.title = 'Condizioni';
32
+ /** Campi disponibili per la selezione nelle condizioni. */
33
+ this.availableFields = [];
34
+ /** Emesso quando la lista di condizioni viene modificata. */
35
+ this.conditionsChange = new EventEmitter();
36
+ /** Indica se l'editor e in modalita JSON. */
37
+ this.jsonMode = false;
38
+ /** Testo JSON corrente nell'editor. */
39
+ this.jsonText = '';
40
+ /** Errore di parsing del JSON. */
41
+ this.jsonError = '';
42
+ /** Opzioni disponibili per gli operatori condizionali. */
43
+ this.operators = [
44
+ { value: 'equals', label: 'Uguale a' },
45
+ { value: 'not_equals', label: 'Diverso da' },
46
+ { value: 'contains', label: 'Contiene' },
47
+ { value: 'not_contains', label: 'Non contiene' },
48
+ { value: 'greater_than', label: 'Maggiore di' },
49
+ { value: 'less_than', label: 'Minore di' },
50
+ { value: 'greater_equal', label: 'Maggiore o uguale' },
51
+ { value: 'less_equal', label: 'Minore o uguale' },
52
+ { value: 'is_empty', label: 'Vuoto' },
53
+ { value: 'is_not_empty', label: 'Non vuoto' },
54
+ { value: 'in_array', label: 'Presente in lista' },
55
+ { value: 'not_in_array', label: 'Non presente in lista' },
56
+ ];
57
+ }
58
+ /**
59
+ * Verifica se l'operatore non richiede un campo valore.
60
+ * @param operator - Operatore da verificare
61
+ * @returns true se l'operatore non necessita di valore
62
+ */
63
+ isNoValueOperator(operator) {
64
+ return operator === 'is_empty' || operator === 'is_not_empty';
65
+ }
66
+ /**
67
+ * Verifica se l'operatore opera su array di valori.
68
+ * @param operator - Operatore da verificare
69
+ * @returns true se l'operatore richiede un array di valori
70
+ */
71
+ isArrayOperator(operator) {
72
+ return operator === 'in_array' || operator === 'not_in_array';
73
+ }
74
+ /**
75
+ * Restituisce le opzioni disponibili per un campo specifico.
76
+ * @param fieldKey - Chiave del campo
77
+ * @returns Array di opzioni del campo, vuoto se non disponibili
78
+ */
79
+ getFieldOptions(fieldKey) {
80
+ const field = this.availableFields.find((f) => f.key === fieldKey);
81
+ return field?.options || [];
82
+ }
83
+ /**
84
+ * Garantisce che il valore sia un array (per select multiple).
85
+ * @param value - Valore da normalizzare
86
+ * @returns Il valore come array
87
+ */
88
+ ensureArray(value) {
89
+ if (Array.isArray(value))
90
+ return value;
91
+ if (value == null || value === '')
92
+ return [];
93
+ return [value];
94
+ }
95
+ /**
96
+ * Attiva/disattiva la modalita editor JSON.
97
+ * In apertura serializza le condizioni, in chiusura le deserializza.
98
+ */
99
+ toggleJsonMode() {
100
+ if (!this.jsonMode) {
101
+ // Passa a modalita JSON: serializza condizioni
102
+ this.jsonText = this.conditions.length > 0 ? JSON.stringify(this.conditions, null, 2) : '[]';
103
+ this.jsonError = '';
104
+ }
105
+ else {
106
+ // Torna a modalita visuale: prova a parsare il JSON
107
+ this.applyJson();
108
+ }
109
+ this.jsonMode = !this.jsonMode;
110
+ }
111
+ /**
112
+ * Formatta il JSON nell'editor con indentazione.
113
+ */
114
+ formatJson() {
115
+ try {
116
+ const parsed = JSON.parse(this.jsonText);
117
+ this.jsonText = JSON.stringify(parsed, null, 2);
118
+ this.jsonError = '';
119
+ }
120
+ catch {
121
+ this.jsonError = 'JSON non valido: impossibile formattare.';
122
+ }
123
+ }
124
+ /**
125
+ * Gestisce il cambio del testo JSON nell'editor.
126
+ * Tenta il parsing e aggiorna le condizioni se valido.
127
+ * @param text - Testo JSON inserito dall'utente
128
+ */
129
+ onJsonChange(text) {
130
+ this.jsonText = text;
131
+ this.applyJson();
132
+ }
133
+ /**
134
+ * Aggiunge una nuova condizione con valori predefiniti.
135
+ */
136
+ addCondition() {
137
+ const firstField = this.availableFields.length > 0 ? this.availableFields[0].key : '';
138
+ const newCondition = {
139
+ field: firstField,
140
+ operator: 'equals',
141
+ value: '',
142
+ };
143
+ const updated = [...this.conditions, newCondition];
144
+ this.emitChange(updated);
145
+ }
146
+ /**
147
+ * Rimuove la condizione all'indice specificato.
148
+ * @param index - Indice della condizione da rimuovere
149
+ */
150
+ removeCondition(index) {
151
+ const updated = this.conditions.filter((_, i) => i !== index);
152
+ this.emitChange(updated);
153
+ }
154
+ /**
155
+ * Rimuove tutte le condizioni.
156
+ */
157
+ clearAll() {
158
+ this.emitChange([]);
159
+ }
160
+ /**
161
+ * Gestisce il cambio della logica di combinazione (AND/OR).
162
+ * @param index - Indice della condizione
163
+ * @param logic - Nuova logica ('AND' | 'OR')
164
+ */
165
+ onLogicChange(index, logic) {
166
+ const updated = this.cloneConditions();
167
+ updated[index] = { ...updated[index], logic };
168
+ this.emitChange(updated);
169
+ }
170
+ /**
171
+ * Gestisce il cambio del campo selezionato in una condizione.
172
+ * Resetta il valore quando il campo cambia.
173
+ * @param index - Indice della condizione
174
+ * @param field - Nuova chiave campo
175
+ */
176
+ onFieldChange(index, field) {
177
+ const updated = this.cloneConditions();
178
+ updated[index] = { ...updated[index], field, value: '' };
179
+ this.emitChange(updated);
180
+ }
181
+ /**
182
+ * Gestisce il cambio dell'operatore in una condizione.
183
+ * Resetta il valore per operatori senza valore.
184
+ * @param index - Indice della condizione
185
+ * @param operator - Nuovo operatore
186
+ */
187
+ onOperatorChange(index, operator) {
188
+ const updated = this.cloneConditions();
189
+ const condition = { ...updated[index], operator };
190
+ // Resetta il valore per operatori che non lo richiedono
191
+ if (this.isNoValueOperator(operator)) {
192
+ condition.value = undefined;
193
+ }
194
+ else if (this.isArrayOperator(operator) && !Array.isArray(condition.value)) {
195
+ // Converti a array per operatori su array
196
+ condition.value = condition.value ? [condition.value] : [];
197
+ }
198
+ updated[index] = condition;
199
+ this.emitChange(updated);
200
+ }
201
+ /**
202
+ * Gestisce il cambio del valore in una condizione.
203
+ * @param index - Indice della condizione
204
+ * @param value - Nuovo valore
205
+ */
206
+ onValueChange(index, value) {
207
+ const updated = this.cloneConditions();
208
+ updated[index] = { ...updated[index], value };
209
+ this.emitChange(updated);
210
+ }
211
+ /**
212
+ * Tenta di applicare il JSON corrente come array di condizioni.
213
+ * Aggiorna l'errore in caso di JSON non valido.
214
+ */
215
+ applyJson() {
216
+ try {
217
+ const parsed = JSON.parse(this.jsonText);
218
+ if (!Array.isArray(parsed)) {
219
+ this.jsonError = 'Il JSON deve essere un array di condizioni.';
220
+ return;
221
+ }
222
+ this.jsonError = '';
223
+ this.emitChange(parsed);
224
+ }
225
+ catch {
226
+ this.jsonError = 'JSON non valido: controlla la sintassi.';
227
+ }
228
+ }
229
+ /**
230
+ * Clona l'array delle condizioni per garantire immutabilita.
231
+ * @returns Copia superficiale dell'array di condizioni
232
+ */
233
+ cloneConditions() {
234
+ return this.conditions.map((c) => ({ ...c }));
235
+ }
236
+ /**
237
+ * Emette l'evento di cambio condizioni.
238
+ * @param conditions - Nuova lista di condizioni
239
+ */
240
+ emitChange(conditions) {
241
+ this.conditions = conditions;
242
+ this.conditionsChange.emit(conditions);
243
+ }
244
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiConditionEditorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
245
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiConditionEditorComponent, isStandalone: true, selector: "ui-condition-editor", inputs: { conditions: "conditions", title: "title", availableFields: "availableFields" }, outputs: { conditionsChange: "conditionsChange" }, ngImport: i0, template: `
246
+ <!-- Intestazione con titolo, statistiche e controlli -->
247
+ <div class="ui-condition-editor__header">
248
+ <div class="ui-condition-editor__header-left">
249
+ <h4 class="ui-condition-editor__title">{{ title }}</h4>
250
+ @if (conditions.length > 0) {
251
+ <span class="ui-condition-editor__badge">{{ conditions.length }}</span>
252
+ }
253
+ </div>
254
+ <div class="ui-condition-editor__header-actions">
255
+ <!-- Pulsante formattazione JSON (solo in modalita JSON) -->
256
+ @if (jsonMode) {
257
+ <ui-button
258
+ icon="braces"
259
+ variant="ghost"
260
+ size="xs"
261
+ tooltip="Formatta JSON"
262
+ ariaLabel="Formatta JSON"
263
+ (click)="formatJson()"
264
+ />
265
+ }
266
+
267
+ <!-- Toggle modalita JSON/visuale -->
268
+ <ui-button
269
+ [icon]="jsonMode ? 'eye' : 'code'"
270
+ [variant]="jsonMode ? 'primary' : 'ghost'"
271
+ size="xs"
272
+ [tooltip]="jsonMode ? 'Editor visuale' : 'Editor JSON'"
273
+ [ariaLabel]="jsonMode ? 'Editor visuale' : 'Editor JSON'"
274
+ (click)="toggleJsonMode()"
275
+ />
276
+
277
+ <!-- Pulsante pulisci tutto -->
278
+ @if (conditions.length > 0) {
279
+ <ui-button
280
+ icon="x"
281
+ variant="warn"
282
+ size="xs"
283
+ tooltip="Rimuovi tutte le condizioni"
284
+ ariaLabel="Rimuovi tutte le condizioni"
285
+ (click)="clearAll()"
286
+ />
287
+ }
288
+ </div>
289
+ </div>
290
+
291
+ <!-- Editor JSON -->
292
+ @if (jsonMode) {
293
+ <div class="ui-condition-editor__json">
294
+ <textarea
295
+ class="ui-condition-editor__json-textarea"
296
+ [ngModel]="jsonText"
297
+ (ngModelChange)="onJsonChange($event)"
298
+ rows="8"
299
+ placeholder='[{ "field": "...", "operator": "equals", "value": "..." }]'
300
+ spellcheck="false"
301
+ ></textarea>
302
+ @if (jsonError) {
303
+ <div class="ui-condition-editor__json-error">
304
+ <lucide-icon name="alert-triangle" [size]="14" />
305
+ <span>{{ jsonError }}</span>
306
+ </div>
307
+ }
308
+ </div>
309
+ } @else {
310
+ <!-- Editor visuale -->
311
+ @if (availableFields.length === 0) {
312
+ <div class="ui-condition-editor__warning">
313
+ <lucide-icon name="alert-triangle" [size]="16" />
314
+ <span>Nessun campo disponibile per le condizioni.</span>
315
+ </div>
316
+ }
317
+
318
+ @if (conditions.length > 0) {
319
+ <div class="ui-condition-editor__list">
320
+ @for (cond of conditions; track $index) {
321
+ <div class="ui-condition-editor__condition-wrapper">
322
+ <!-- Selettore logica AND/OR (dal secondo elemento) -->
323
+ @if ($index > 0) {
324
+ <div class="ui-condition-editor__logic-row">
325
+ <ui-button
326
+ label="AND"
327
+ [variant]="(cond.logic || 'AND') === 'AND' ? 'primary' : 'outline'"
328
+ size="xs"
329
+ (click)="onLogicChange($index, 'AND')"
330
+ />
331
+ <ui-button
332
+ label="OR"
333
+ [variant]="cond.logic === 'OR' ? 'primary' : 'outline'"
334
+ size="xs"
335
+ (click)="onLogicChange($index, 'OR')"
336
+ />
337
+ </div>
338
+ }
339
+
340
+ <mat-card class="ui-condition-editor__card" appearance="outlined">
341
+ <mat-card-content>
342
+ <div class="ui-condition-editor__card-grid">
343
+ <!-- Selezione campo -->
344
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
345
+ <mat-label>Campo</mat-label>
346
+ <mat-select
347
+ [ngModel]="cond.field"
348
+ (ngModelChange)="onFieldChange($index, $event)"
349
+ >
350
+ @for (f of availableFields; track f.key) {
351
+ <mat-option [value]="f.key">{{ f.label }}</mat-option>
352
+ }
353
+ </mat-select>
354
+ </mat-form-field>
355
+
356
+ <!-- Selezione operatore -->
357
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
358
+ <mat-label>Operatore</mat-label>
359
+ <mat-select
360
+ [ngModel]="cond.operator"
361
+ (ngModelChange)="onOperatorChange($index, $event)"
362
+ >
363
+ @for (op of operators; track op.value) {
364
+ <mat-option [value]="op.value">{{ op.label }}</mat-option>
365
+ }
366
+ </mat-select>
367
+ </mat-form-field>
368
+
369
+ <!-- Campo valore (se l'operatore lo richiede) -->
370
+ @if (!isNoValueOperator(cond.operator)) {
371
+ <!-- Select multipla per in_array/not_in_array con opzioni disponibili -->
372
+ @if (isArrayOperator(cond.operator) && getFieldOptions(cond.field).length > 0) {
373
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
374
+ <mat-label>Valori</mat-label>
375
+ <mat-select
376
+ [ngModel]="ensureArray(cond.value)"
377
+ (ngModelChange)="onValueChange($index, $event)"
378
+ multiple
379
+ >
380
+ @for (opt of getFieldOptions(cond.field); track opt.value) {
381
+ <mat-option [value]="opt.value">{{ opt.label }}</mat-option>
382
+ }
383
+ </mat-select>
384
+ </mat-form-field>
385
+ } @else if (getFieldOptions(cond.field).length > 0 && !isArrayOperator(cond.operator)) {
386
+ <!-- Select singola se il campo ha opzioni -->
387
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
388
+ <mat-label>Valore</mat-label>
389
+ <mat-select
390
+ [ngModel]="cond.value"
391
+ (ngModelChange)="onValueChange($index, $event)"
392
+ >
393
+ @for (opt of getFieldOptions(cond.field); track opt.value) {
394
+ <mat-option [value]="opt.value">{{ opt.label }}</mat-option>
395
+ }
396
+ </mat-select>
397
+ </mat-form-field>
398
+ } @else {
399
+ <!-- Input testo libero -->
400
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
401
+ <mat-label>Valore</mat-label>
402
+ <input
403
+ matInput
404
+ [ngModel]="cond.value"
405
+ (ngModelChange)="onValueChange($index, $event)"
406
+ placeholder="Inserisci valore"
407
+ />
408
+ </mat-form-field>
409
+ }
410
+ }
411
+
412
+ <!-- Pulsante rimuovi -->
413
+ <ui-button
414
+ icon="trash-2"
415
+ variant="warn"
416
+ size="xs"
417
+ tooltip="Rimuovi condizione"
418
+ ariaLabel="Rimuovi condizione"
419
+ (click)="removeCondition($index)"
420
+ />
421
+ </div>
422
+ </mat-card-content>
423
+ </mat-card>
424
+ </div>
425
+ }
426
+ </div>
427
+ } @else {
428
+ <!-- Stato vuoto (editor visuale) -->
429
+ <div class="ui-condition-editor__empty">
430
+ <lucide-icon name="git-branch" [size]="28" class="ui-condition-editor__empty-icon" />
431
+ <p class="ui-condition-editor__empty-text">Nessuna condizione configurata.</p>
432
+ </div>
433
+ }
434
+
435
+ <!-- Pulsante aggiungi condizione -->
436
+ <ui-button
437
+ label="Aggiungi condizione"
438
+ icon="plus"
439
+ variant="outline"
440
+ size="sm"
441
+ [disabled]="availableFields.length === 0"
442
+ (click)="addCondition()"
443
+ />
444
+ }
445
+ `, isInline: true, styles: [":host{display:block}.ui-condition-editor__header{display:flex;align-items:center;justify-content:space-between;margin-bottom:var(--ui-spacing-3)}.ui-condition-editor__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2)}.ui-condition-editor__title{margin:0;font-size:var(--ui-font-size-sm);font-weight:600;color:var(--ui-color-text)}.ui-condition-editor__badge{display:inline-flex;align-items:center;justify-content:center;min-width:20px;height:20px;padding:0 var(--ui-spacing-1);border-radius:10px;background:var(--ui-color-primary);color:var(--ui-color-surface);font-size:var(--ui-font-size-xs);font-weight:600;line-height:1}.ui-condition-editor__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1)}.ui-condition-editor__json{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-condition-editor__json-textarea{width:100%;min-height:120px;padding:var(--ui-spacing-3);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-bg-subtle);color:var(--ui-color-text);font-family:Fira Code,Consolas,monospace;font-size:var(--ui-font-size-xs);line-height:1.5;resize:vertical;box-sizing:border-box;&:focus{outline:none;border-color:var(--ui-color-primary);box-shadow:0 0 0 2px color-mix(in srgb,var(--ui-color-primary) 20%,transparent)}}.ui-condition-editor__json-error{display:flex;align-items:center;gap:var(--ui-spacing-1);padding:var(--ui-spacing-2);border-radius:var(--ui-radius-sm);background:color-mix(in srgb,var(--ui-color-warn) 10%,transparent);color:var(--ui-color-warn);font-size:var(--ui-font-size-xs)}.ui-condition-editor__warning{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3);margin-bottom:var(--ui-spacing-3);border-radius:var(--ui-radius-sm);background:color-mix(in srgb,var(--ui-color-warn) 10%,transparent);color:var(--ui-color-warn);font-size:var(--ui-font-size-xs)}.ui-condition-editor__list{display:flex;flex-direction:column;gap:0;margin-bottom:var(--ui-spacing-3)}.ui-condition-editor__condition-wrapper{display:flex;flex-direction:column}.ui-condition-editor__logic-row{display:flex;align-items:center;justify-content:center;gap:var(--ui-spacing-1);padding:var(--ui-spacing-2) 0}.ui-condition-editor__card{background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-sm)}.ui-condition-editor__card-grid{display:flex;align-items:flex-start;gap:var(--ui-spacing-2);flex-wrap:wrap}.ui-condition-editor__field{flex:1;min-width:140px}.ui-condition-editor__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:var(--ui-spacing-5) var(--ui-spacing-4);margin-bottom:var(--ui-spacing-3);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-bg-subtle);text-align:center}.ui-condition-editor__empty-icon{color:var(--ui-color-text-muted);margin-bottom:var(--ui-spacing-2)}.ui-condition-editor__empty-text{margin:0;font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary)}\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: "directive", type: i2.MatCardContent, selector: "mat-card-content" }, { 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: MatSelectModule }, { kind: "component", type: i5.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i6.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { 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 }); }
446
+ }
447
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiConditionEditorComponent, decorators: [{
448
+ type: Component,
449
+ args: [{ selector: 'ui-condition-editor', standalone: true, imports: [
450
+ FormsModule,
451
+ MatCardModule,
452
+ MatFormFieldModule,
453
+ MatInputModule,
454
+ MatSelectModule,
455
+ LucideAngularModule,
456
+ UiButtonComponent,
457
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: `
458
+ <!-- Intestazione con titolo, statistiche e controlli -->
459
+ <div class="ui-condition-editor__header">
460
+ <div class="ui-condition-editor__header-left">
461
+ <h4 class="ui-condition-editor__title">{{ title }}</h4>
462
+ @if (conditions.length > 0) {
463
+ <span class="ui-condition-editor__badge">{{ conditions.length }}</span>
464
+ }
465
+ </div>
466
+ <div class="ui-condition-editor__header-actions">
467
+ <!-- Pulsante formattazione JSON (solo in modalita JSON) -->
468
+ @if (jsonMode) {
469
+ <ui-button
470
+ icon="braces"
471
+ variant="ghost"
472
+ size="xs"
473
+ tooltip="Formatta JSON"
474
+ ariaLabel="Formatta JSON"
475
+ (click)="formatJson()"
476
+ />
477
+ }
478
+
479
+ <!-- Toggle modalita JSON/visuale -->
480
+ <ui-button
481
+ [icon]="jsonMode ? 'eye' : 'code'"
482
+ [variant]="jsonMode ? 'primary' : 'ghost'"
483
+ size="xs"
484
+ [tooltip]="jsonMode ? 'Editor visuale' : 'Editor JSON'"
485
+ [ariaLabel]="jsonMode ? 'Editor visuale' : 'Editor JSON'"
486
+ (click)="toggleJsonMode()"
487
+ />
488
+
489
+ <!-- Pulsante pulisci tutto -->
490
+ @if (conditions.length > 0) {
491
+ <ui-button
492
+ icon="x"
493
+ variant="warn"
494
+ size="xs"
495
+ tooltip="Rimuovi tutte le condizioni"
496
+ ariaLabel="Rimuovi tutte le condizioni"
497
+ (click)="clearAll()"
498
+ />
499
+ }
500
+ </div>
501
+ </div>
502
+
503
+ <!-- Editor JSON -->
504
+ @if (jsonMode) {
505
+ <div class="ui-condition-editor__json">
506
+ <textarea
507
+ class="ui-condition-editor__json-textarea"
508
+ [ngModel]="jsonText"
509
+ (ngModelChange)="onJsonChange($event)"
510
+ rows="8"
511
+ placeholder='[{ "field": "...", "operator": "equals", "value": "..." }]'
512
+ spellcheck="false"
513
+ ></textarea>
514
+ @if (jsonError) {
515
+ <div class="ui-condition-editor__json-error">
516
+ <lucide-icon name="alert-triangle" [size]="14" />
517
+ <span>{{ jsonError }}</span>
518
+ </div>
519
+ }
520
+ </div>
521
+ } @else {
522
+ <!-- Editor visuale -->
523
+ @if (availableFields.length === 0) {
524
+ <div class="ui-condition-editor__warning">
525
+ <lucide-icon name="alert-triangle" [size]="16" />
526
+ <span>Nessun campo disponibile per le condizioni.</span>
527
+ </div>
528
+ }
529
+
530
+ @if (conditions.length > 0) {
531
+ <div class="ui-condition-editor__list">
532
+ @for (cond of conditions; track $index) {
533
+ <div class="ui-condition-editor__condition-wrapper">
534
+ <!-- Selettore logica AND/OR (dal secondo elemento) -->
535
+ @if ($index > 0) {
536
+ <div class="ui-condition-editor__logic-row">
537
+ <ui-button
538
+ label="AND"
539
+ [variant]="(cond.logic || 'AND') === 'AND' ? 'primary' : 'outline'"
540
+ size="xs"
541
+ (click)="onLogicChange($index, 'AND')"
542
+ />
543
+ <ui-button
544
+ label="OR"
545
+ [variant]="cond.logic === 'OR' ? 'primary' : 'outline'"
546
+ size="xs"
547
+ (click)="onLogicChange($index, 'OR')"
548
+ />
549
+ </div>
550
+ }
551
+
552
+ <mat-card class="ui-condition-editor__card" appearance="outlined">
553
+ <mat-card-content>
554
+ <div class="ui-condition-editor__card-grid">
555
+ <!-- Selezione campo -->
556
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
557
+ <mat-label>Campo</mat-label>
558
+ <mat-select
559
+ [ngModel]="cond.field"
560
+ (ngModelChange)="onFieldChange($index, $event)"
561
+ >
562
+ @for (f of availableFields; track f.key) {
563
+ <mat-option [value]="f.key">{{ f.label }}</mat-option>
564
+ }
565
+ </mat-select>
566
+ </mat-form-field>
567
+
568
+ <!-- Selezione operatore -->
569
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
570
+ <mat-label>Operatore</mat-label>
571
+ <mat-select
572
+ [ngModel]="cond.operator"
573
+ (ngModelChange)="onOperatorChange($index, $event)"
574
+ >
575
+ @for (op of operators; track op.value) {
576
+ <mat-option [value]="op.value">{{ op.label }}</mat-option>
577
+ }
578
+ </mat-select>
579
+ </mat-form-field>
580
+
581
+ <!-- Campo valore (se l'operatore lo richiede) -->
582
+ @if (!isNoValueOperator(cond.operator)) {
583
+ <!-- Select multipla per in_array/not_in_array con opzioni disponibili -->
584
+ @if (isArrayOperator(cond.operator) && getFieldOptions(cond.field).length > 0) {
585
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
586
+ <mat-label>Valori</mat-label>
587
+ <mat-select
588
+ [ngModel]="ensureArray(cond.value)"
589
+ (ngModelChange)="onValueChange($index, $event)"
590
+ multiple
591
+ >
592
+ @for (opt of getFieldOptions(cond.field); track opt.value) {
593
+ <mat-option [value]="opt.value">{{ opt.label }}</mat-option>
594
+ }
595
+ </mat-select>
596
+ </mat-form-field>
597
+ } @else if (getFieldOptions(cond.field).length > 0 && !isArrayOperator(cond.operator)) {
598
+ <!-- Select singola se il campo ha opzioni -->
599
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
600
+ <mat-label>Valore</mat-label>
601
+ <mat-select
602
+ [ngModel]="cond.value"
603
+ (ngModelChange)="onValueChange($index, $event)"
604
+ >
605
+ @for (opt of getFieldOptions(cond.field); track opt.value) {
606
+ <mat-option [value]="opt.value">{{ opt.label }}</mat-option>
607
+ }
608
+ </mat-select>
609
+ </mat-form-field>
610
+ } @else {
611
+ <!-- Input testo libero -->
612
+ <mat-form-field class="ui-condition-editor__field" appearance="outline">
613
+ <mat-label>Valore</mat-label>
614
+ <input
615
+ matInput
616
+ [ngModel]="cond.value"
617
+ (ngModelChange)="onValueChange($index, $event)"
618
+ placeholder="Inserisci valore"
619
+ />
620
+ </mat-form-field>
621
+ }
622
+ }
623
+
624
+ <!-- Pulsante rimuovi -->
625
+ <ui-button
626
+ icon="trash-2"
627
+ variant="warn"
628
+ size="xs"
629
+ tooltip="Rimuovi condizione"
630
+ ariaLabel="Rimuovi condizione"
631
+ (click)="removeCondition($index)"
632
+ />
633
+ </div>
634
+ </mat-card-content>
635
+ </mat-card>
636
+ </div>
637
+ }
638
+ </div>
639
+ } @else {
640
+ <!-- Stato vuoto (editor visuale) -->
641
+ <div class="ui-condition-editor__empty">
642
+ <lucide-icon name="git-branch" [size]="28" class="ui-condition-editor__empty-icon" />
643
+ <p class="ui-condition-editor__empty-text">Nessuna condizione configurata.</p>
644
+ </div>
645
+ }
646
+
647
+ <!-- Pulsante aggiungi condizione -->
648
+ <ui-button
649
+ label="Aggiungi condizione"
650
+ icon="plus"
651
+ variant="outline"
652
+ size="sm"
653
+ [disabled]="availableFields.length === 0"
654
+ (click)="addCondition()"
655
+ />
656
+ }
657
+ `, styles: [":host{display:block}.ui-condition-editor__header{display:flex;align-items:center;justify-content:space-between;margin-bottom:var(--ui-spacing-3)}.ui-condition-editor__header-left{display:flex;align-items:center;gap:var(--ui-spacing-2)}.ui-condition-editor__title{margin:0;font-size:var(--ui-font-size-sm);font-weight:600;color:var(--ui-color-text)}.ui-condition-editor__badge{display:inline-flex;align-items:center;justify-content:center;min-width:20px;height:20px;padding:0 var(--ui-spacing-1);border-radius:10px;background:var(--ui-color-primary);color:var(--ui-color-surface);font-size:var(--ui-font-size-xs);font-weight:600;line-height:1}.ui-condition-editor__header-actions{display:flex;align-items:center;gap:var(--ui-spacing-1)}.ui-condition-editor__json{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-condition-editor__json-textarea{width:100%;min-height:120px;padding:var(--ui-spacing-3);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-bg-subtle);color:var(--ui-color-text);font-family:Fira Code,Consolas,monospace;font-size:var(--ui-font-size-xs);line-height:1.5;resize:vertical;box-sizing:border-box;&:focus{outline:none;border-color:var(--ui-color-primary);box-shadow:0 0 0 2px color-mix(in srgb,var(--ui-color-primary) 20%,transparent)}}.ui-condition-editor__json-error{display:flex;align-items:center;gap:var(--ui-spacing-1);padding:var(--ui-spacing-2);border-radius:var(--ui-radius-sm);background:color-mix(in srgb,var(--ui-color-warn) 10%,transparent);color:var(--ui-color-warn);font-size:var(--ui-font-size-xs)}.ui-condition-editor__warning{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3);margin-bottom:var(--ui-spacing-3);border-radius:var(--ui-radius-sm);background:color-mix(in srgb,var(--ui-color-warn) 10%,transparent);color:var(--ui-color-warn);font-size:var(--ui-font-size-xs)}.ui-condition-editor__list{display:flex;flex-direction:column;gap:0;margin-bottom:var(--ui-spacing-3)}.ui-condition-editor__condition-wrapper{display:flex;flex-direction:column}.ui-condition-editor__logic-row{display:flex;align-items:center;justify-content:center;gap:var(--ui-spacing-1);padding:var(--ui-spacing-2) 0}.ui-condition-editor__card{background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-sm)}.ui-condition-editor__card-grid{display:flex;align-items:flex-start;gap:var(--ui-spacing-2);flex-wrap:wrap}.ui-condition-editor__field{flex:1;min-width:140px}.ui-condition-editor__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:var(--ui-spacing-5) var(--ui-spacing-4);margin-bottom:var(--ui-spacing-3);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-bg-subtle);text-align:center}.ui-condition-editor__empty-icon{color:var(--ui-color-text-muted);margin-bottom:var(--ui-spacing-2)}.ui-condition-editor__empty-text{margin:0;font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary)}\n"] }]
658
+ }], propDecorators: { conditions: [{
659
+ type: Input
660
+ }], title: [{
661
+ type: Input
662
+ }], availableFields: [{
663
+ type: Input
664
+ }], conditionsChange: [{
665
+ type: Output
666
+ }] } });
667
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"condition-editor.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy } 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 { MatSelectModule } from '@angular/material/select';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiButtonComponent } from '../../../button/index';\r\nimport { UiFieldCondition, UiConditionalOperator } from '../../../form-builder/types/index';\r\nimport { UiEditorAvailableField } from '../../types/editor.types';\r\n\r\n/**\r\n * Interfaccia per le opzioni degli operatori condizionali nella select.\r\n */\r\ninterface OperatorOption {\r\n  /** Valore dell'operatore. */\r\n  value: UiConditionalOperator;\r\n  /** Etichetta visualizzata nella select. */\r\n  label: string;\r\n}\r\n\r\n/**\r\n * Editor per le condizioni di visibilita/disabilitazione dei campi.\r\n *\r\n * Permette di configurare condizioni con editor visuale o JSON.\r\n * Supporta operatori di confronto multipli, logica AND/OR\r\n * e selezione dei valori dalle opzioni dei campi disponibili.\r\n *\r\n * @selector ui-condition-editor\r\n */\r\n@Component({\r\n  selector: 'ui-condition-editor',\r\n  standalone: true,\r\n  imports: [\r\n    FormsModule,\r\n    MatCardModule,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    LucideAngularModule,\r\n    UiButtonComponent,\r\n  ],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <!-- Intestazione con titolo, statistiche e controlli -->\r\n    <div class=\"ui-condition-editor__header\">\r\n      <div class=\"ui-condition-editor__header-left\">\r\n        <h4 class=\"ui-condition-editor__title\">{{ title }}</h4>\r\n        @if (conditions.length > 0) {\r\n          <span class=\"ui-condition-editor__badge\">{{ conditions.length }}</span>\r\n        }\r\n      </div>\r\n      <div class=\"ui-condition-editor__header-actions\">\r\n        <!-- Pulsante formattazione JSON (solo in modalita JSON) -->\r\n        @if (jsonMode) {\r\n          <ui-button\r\n            icon=\"braces\"\r\n            variant=\"ghost\"\r\n            size=\"xs\"\r\n            tooltip=\"Formatta JSON\"\r\n            ariaLabel=\"Formatta JSON\"\r\n            (click)=\"formatJson()\"\r\n          />\r\n        }\r\n\r\n        <!-- Toggle modalita JSON/visuale -->\r\n        <ui-button\r\n          [icon]=\"jsonMode ? 'eye' : 'code'\"\r\n          [variant]=\"jsonMode ? 'primary' : 'ghost'\"\r\n          size=\"xs\"\r\n          [tooltip]=\"jsonMode ? 'Editor visuale' : 'Editor JSON'\"\r\n          [ariaLabel]=\"jsonMode ? 'Editor visuale' : 'Editor JSON'\"\r\n          (click)=\"toggleJsonMode()\"\r\n        />\r\n\r\n        <!-- Pulsante pulisci tutto -->\r\n        @if (conditions.length > 0) {\r\n          <ui-button\r\n            icon=\"x\"\r\n            variant=\"warn\"\r\n            size=\"xs\"\r\n            tooltip=\"Rimuovi tutte le condizioni\"\r\n            ariaLabel=\"Rimuovi tutte le condizioni\"\r\n            (click)=\"clearAll()\"\r\n          />\r\n        }\r\n      </div>\r\n    </div>\r\n\r\n    <!-- Editor JSON -->\r\n    @if (jsonMode) {\r\n      <div class=\"ui-condition-editor__json\">\r\n        <textarea\r\n          class=\"ui-condition-editor__json-textarea\"\r\n          [ngModel]=\"jsonText\"\r\n          (ngModelChange)=\"onJsonChange($event)\"\r\n          rows=\"8\"\r\n          placeholder='[{ \"field\": \"...\", \"operator\": \"equals\", \"value\": \"...\" }]'\r\n          spellcheck=\"false\"\r\n        ></textarea>\r\n        @if (jsonError) {\r\n          <div class=\"ui-condition-editor__json-error\">\r\n            <lucide-icon name=\"alert-triangle\" [size]=\"14\" />\r\n            <span>{{ jsonError }}</span>\r\n          </div>\r\n        }\r\n      </div>\r\n    } @else {\r\n      <!-- Editor visuale -->\r\n      @if (availableFields.length === 0) {\r\n        <div class=\"ui-condition-editor__warning\">\r\n          <lucide-icon name=\"alert-triangle\" [size]=\"16\" />\r\n          <span>Nessun campo disponibile per le condizioni.</span>\r\n        </div>\r\n      }\r\n\r\n      @if (conditions.length > 0) {\r\n        <div class=\"ui-condition-editor__list\">\r\n          @for (cond of conditions; track $index) {\r\n            <div class=\"ui-condition-editor__condition-wrapper\">\r\n              <!-- Selettore logica AND/OR (dal secondo elemento) -->\r\n              @if ($index > 0) {\r\n                <div class=\"ui-condition-editor__logic-row\">\r\n                  <ui-button\r\n                    label=\"AND\"\r\n                    [variant]=\"(cond.logic || 'AND') === 'AND' ? 'primary' : 'outline'\"\r\n                    size=\"xs\"\r\n                    (click)=\"onLogicChange($index, 'AND')\"\r\n                  />\r\n                  <ui-button\r\n                    label=\"OR\"\r\n                    [variant]=\"cond.logic === 'OR' ? 'primary' : 'outline'\"\r\n                    size=\"xs\"\r\n                    (click)=\"onLogicChange($index, 'OR')\"\r\n                  />\r\n                </div>\r\n              }\r\n\r\n              <mat-card class=\"ui-condition-editor__card\" appearance=\"outlined\">\r\n                <mat-card-content>\r\n                  <div class=\"ui-condition-editor__card-grid\">\r\n                    <!-- Selezione campo -->\r\n                    <mat-form-field class=\"ui-condition-editor__field\" appearance=\"outline\">\r\n                      <mat-label>Campo</mat-label>\r\n                      <mat-select\r\n                        [ngModel]=\"cond.field\"\r\n                        (ngModelChange)=\"onFieldChange($index, $event)\"\r\n                      >\r\n                        @for (f of availableFields; track f.key) {\r\n                          <mat-option [value]=\"f.key\">{{ f.label }}</mat-option>\r\n                        }\r\n                      </mat-select>\r\n                    </mat-form-field>\r\n\r\n                    <!-- Selezione operatore -->\r\n                    <mat-form-field class=\"ui-condition-editor__field\" appearance=\"outline\">\r\n                      <mat-label>Operatore</mat-label>\r\n                      <mat-select\r\n                        [ngModel]=\"cond.operator\"\r\n                        (ngModelChange)=\"onOperatorChange($index, $event)\"\r\n                      >\r\n                        @for (op of operators; track op.value) {\r\n                          <mat-option [value]=\"op.value\">{{ op.label }}</mat-option>\r\n                        }\r\n                      </mat-select>\r\n                    </mat-form-field>\r\n\r\n                    <!-- Campo valore (se l'operatore lo richiede) -->\r\n                    @if (!isNoValueOperator(cond.operator)) {\r\n                      <!-- Select multipla per in_array/not_in_array con opzioni disponibili -->\r\n                      @if (isArrayOperator(cond.operator) && getFieldOptions(cond.field).length > 0) {\r\n                        <mat-form-field class=\"ui-condition-editor__field\" appearance=\"outline\">\r\n                          <mat-label>Valori</mat-label>\r\n                          <mat-select\r\n                            [ngModel]=\"ensureArray(cond.value)\"\r\n                            (ngModelChange)=\"onValueChange($index, $event)\"\r\n                            multiple\r\n                          >\r\n                            @for (opt of getFieldOptions(cond.field); track opt.value) {\r\n                              <mat-option [value]=\"opt.value\">{{ opt.label }}</mat-option>\r\n                            }\r\n                          </mat-select>\r\n                        </mat-form-field>\r\n                      } @else if (getFieldOptions(cond.field).length > 0 && !isArrayOperator(cond.operator)) {\r\n                        <!-- Select singola se il campo ha opzioni -->\r\n                        <mat-form-field class=\"ui-condition-editor__field\" appearance=\"outline\">\r\n                          <mat-label>Valore</mat-label>\r\n                          <mat-select\r\n                            [ngModel]=\"cond.value\"\r\n                            (ngModelChange)=\"onValueChange($index, $event)\"\r\n                          >\r\n                            @for (opt of getFieldOptions(cond.field); track opt.value) {\r\n                              <mat-option [value]=\"opt.value\">{{ opt.label }}</mat-option>\r\n                            }\r\n                          </mat-select>\r\n                        </mat-form-field>\r\n                      } @else {\r\n                        <!-- Input testo libero -->\r\n                        <mat-form-field class=\"ui-condition-editor__field\" appearance=\"outline\">\r\n                          <mat-label>Valore</mat-label>\r\n                          <input\r\n                            matInput\r\n                            [ngModel]=\"cond.value\"\r\n                            (ngModelChange)=\"onValueChange($index, $event)\"\r\n                            placeholder=\"Inserisci valore\"\r\n                          />\r\n                        </mat-form-field>\r\n                      }\r\n                    }\r\n\r\n                    <!-- Pulsante rimuovi -->\r\n                    <ui-button\r\n                      icon=\"trash-2\"\r\n                      variant=\"warn\"\r\n                      size=\"xs\"\r\n                      tooltip=\"Rimuovi condizione\"\r\n                      ariaLabel=\"Rimuovi condizione\"\r\n                      (click)=\"removeCondition($index)\"\r\n                    />\r\n                  </div>\r\n                </mat-card-content>\r\n              </mat-card>\r\n            </div>\r\n          }\r\n        </div>\r\n      } @else {\r\n        <!-- Stato vuoto (editor visuale) -->\r\n        <div class=\"ui-condition-editor__empty\">\r\n          <lucide-icon name=\"git-branch\" [size]=\"28\" class=\"ui-condition-editor__empty-icon\" />\r\n          <p class=\"ui-condition-editor__empty-text\">Nessuna condizione configurata.</p>\r\n        </div>\r\n      }\r\n\r\n      <!-- Pulsante aggiungi condizione -->\r\n      <ui-button\r\n        label=\"Aggiungi condizione\"\r\n        icon=\"plus\"\r\n        variant=\"outline\"\r\n        size=\"sm\"\r\n        [disabled]=\"availableFields.length === 0\"\r\n        (click)=\"addCondition()\"\r\n      />\r\n    }\r\n  `,\r\n  styles: [\r\n    `\r\n    :host {\r\n      display: block;\r\n    }\r\n\r\n    .ui-condition-editor__header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      margin-bottom: var(--ui-spacing-3);\r\n    }\r\n\r\n    .ui-condition-editor__header-left {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n    }\r\n\r\n    .ui-condition-editor__title {\r\n      margin: 0;\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    .ui-condition-editor__badge {\r\n      display: inline-flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      min-width: 20px;\r\n      height: 20px;\r\n      padding: 0 var(--ui-spacing-1);\r\n      border-radius: 10px;\r\n      background: var(--ui-color-primary);\r\n      color: var(--ui-color-surface);\r\n      font-size: var(--ui-font-size-xs);\r\n      font-weight: 600;\r\n      line-height: 1;\r\n    }\r\n\r\n    .ui-condition-editor__header-actions {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1);\r\n    }\r\n\r\n    /* --- Editor JSON --- */\r\n\r\n    .ui-condition-editor__json {\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: var(--ui-spacing-2);\r\n    }\r\n\r\n    .ui-condition-editor__json-textarea {\r\n      width: 100%;\r\n      min-height: 120px;\r\n      padding: var(--ui-spacing-3);\r\n      border: 1px solid var(--ui-color-border);\r\n      border-radius: var(--ui-radius-md);\r\n      background: var(--ui-color-bg-subtle);\r\n      color: var(--ui-color-text);\r\n      font-family: 'Fira Code', 'Consolas', monospace;\r\n      font-size: var(--ui-font-size-xs);\r\n      line-height: 1.5;\r\n      resize: vertical;\r\n      box-sizing: border-box;\r\n\r\n      &:focus {\r\n        outline: none;\r\n        border-color: var(--ui-color-primary);\r\n        box-shadow: 0 0 0 2px color-mix(in srgb, var(--ui-color-primary) 20%, transparent);\r\n      }\r\n    }\r\n\r\n    .ui-condition-editor__json-error {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-1);\r\n      padding: var(--ui-spacing-2);\r\n      border-radius: var(--ui-radius-sm);\r\n      background: color-mix(in srgb, var(--ui-color-warn) 10%, transparent);\r\n      color: var(--ui-color-warn);\r\n      font-size: var(--ui-font-size-xs);\r\n    }\r\n\r\n    /* --- Avviso nessun campo disponibile --- */\r\n\r\n    .ui-condition-editor__warning {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2);\r\n      padding: var(--ui-spacing-3);\r\n      margin-bottom: var(--ui-spacing-3);\r\n      border-radius: var(--ui-radius-sm);\r\n      background: color-mix(in srgb, var(--ui-color-warn) 10%, transparent);\r\n      color: var(--ui-color-warn);\r\n      font-size: var(--ui-font-size-xs);\r\n    }\r\n\r\n    /* --- Lista condizioni --- */\r\n\r\n    .ui-condition-editor__list {\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: 0;\r\n      margin-bottom: var(--ui-spacing-3);\r\n    }\r\n\r\n    .ui-condition-editor__condition-wrapper {\r\n      display: flex;\r\n      flex-direction: column;\r\n    }\r\n\r\n    .ui-condition-editor__logic-row {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      gap: var(--ui-spacing-1);\r\n      padding: var(--ui-spacing-2) 0;\r\n    }\r\n\r\n    .ui-condition-editor__card {\r\n      background: var(--ui-color-surface);\r\n      border: 1px solid var(--ui-color-border);\r\n      border-radius: var(--ui-radius-md);\r\n      box-shadow: var(--ui-shadow-sm);\r\n    }\r\n\r\n    .ui-condition-editor__card-grid {\r\n      display: flex;\r\n      align-items: flex-start;\r\n      gap: var(--ui-spacing-2);\r\n      flex-wrap: wrap;\r\n    }\r\n\r\n    .ui-condition-editor__field {\r\n      flex: 1;\r\n      min-width: 140px;\r\n    }\r\n\r\n    /* --- Stato vuoto --- */\r\n\r\n    .ui-condition-editor__empty {\r\n      display: flex;\r\n      flex-direction: column;\r\n      align-items: center;\r\n      justify-content: center;\r\n      padding: var(--ui-spacing-5) var(--ui-spacing-4);\r\n      margin-bottom: var(--ui-spacing-3);\r\n      border: 1px dashed var(--ui-color-border);\r\n      border-radius: var(--ui-radius-md);\r\n      background: var(--ui-color-bg-subtle);\r\n      text-align: center;\r\n    }\r\n\r\n    .ui-condition-editor__empty-icon {\r\n      color: var(--ui-color-text-muted);\r\n      margin-bottom: var(--ui-spacing-2);\r\n    }\r\n\r\n    .ui-condition-editor__empty-text {\r\n      margin: 0;\r\n      font-size: var(--ui-font-size-sm);\r\n      color: var(--ui-color-text-secondary);\r\n    }\r\n\r\n  `,\r\n  ],\r\n})\r\nexport class UiConditionEditorComponent {\r\n  /** Lista delle condizioni da editare. */\r\n  @Input() conditions: UiFieldCondition[] = [];\r\n\r\n  /** Titolo visualizzato nell'intestazione. */\r\n  @Input() title = 'Condizioni';\r\n\r\n  /** Campi disponibili per la selezione nelle condizioni. */\r\n  @Input() availableFields: UiEditorAvailableField[] = [];\r\n\r\n  /** Emesso quando la lista di condizioni viene modificata. */\r\n  @Output() conditionsChange = new EventEmitter<UiFieldCondition[]>();\r\n\r\n  /** Indica se l'editor e in modalita JSON. */\r\n  jsonMode = false;\r\n\r\n  /** Testo JSON corrente nell'editor. */\r\n  jsonText = '';\r\n\r\n  /** Errore di parsing del JSON. */\r\n  jsonError = '';\r\n\r\n  /** Opzioni disponibili per gli operatori condizionali. */\r\n  readonly operators: OperatorOption[] = [\r\n    { value: 'equals', label: 'Uguale a' },\r\n    { value: 'not_equals', label: 'Diverso da' },\r\n    { value: 'contains', label: 'Contiene' },\r\n    { value: 'not_contains', label: 'Non contiene' },\r\n    { value: 'greater_than', label: 'Maggiore di' },\r\n    { value: 'less_than', label: 'Minore di' },\r\n    { value: 'greater_equal', label: 'Maggiore o uguale' },\r\n    { value: 'less_equal', label: 'Minore o uguale' },\r\n    { value: 'is_empty', label: 'Vuoto' },\r\n    { value: 'is_not_empty', label: 'Non vuoto' },\r\n    { value: 'in_array', label: 'Presente in lista' },\r\n    { value: 'not_in_array', label: 'Non presente in lista' },\r\n  ];\r\n\r\n  /**\r\n   * Verifica se l'operatore non richiede un campo valore.\r\n   * @param operator - Operatore da verificare\r\n   * @returns true se l'operatore non necessita di valore\r\n   */\r\n  isNoValueOperator(operator: UiConditionalOperator): boolean {\r\n    return operator === 'is_empty' || operator === 'is_not_empty';\r\n  }\r\n\r\n  /**\r\n   * Verifica se l'operatore opera su array di valori.\r\n   * @param operator - Operatore da verificare\r\n   * @returns true se l'operatore richiede un array di valori\r\n   */\r\n  isArrayOperator(operator: UiConditionalOperator): boolean {\r\n    return operator === 'in_array' || operator === 'not_in_array';\r\n  }\r\n\r\n  /**\r\n   * Restituisce le opzioni disponibili per un campo specifico.\r\n   * @param fieldKey - Chiave del campo\r\n   * @returns Array di opzioni del campo, vuoto se non disponibili\r\n   */\r\n  getFieldOptions(fieldKey: string): { value: any; label: string }[] {\r\n    const field = this.availableFields.find((f) => f.key === fieldKey);\r\n    return field?.options || [];\r\n  }\r\n\r\n  /**\r\n   * Garantisce che il valore sia un array (per select multiple).\r\n   * @param value - Valore da normalizzare\r\n   * @returns Il valore come array\r\n   */\r\n  ensureArray(value: any): any[] {\r\n    if (Array.isArray(value)) return value;\r\n    if (value == null || value === '') return [];\r\n    return [value];\r\n  }\r\n\r\n  /**\r\n   * Attiva/disattiva la modalita editor JSON.\r\n   * In apertura serializza le condizioni, in chiusura le deserializza.\r\n   */\r\n  toggleJsonMode(): void {\r\n    if (!this.jsonMode) {\r\n      // Passa a modalita JSON: serializza condizioni\r\n      this.jsonText = this.conditions.length > 0 ? JSON.stringify(this.conditions, null, 2) : '[]';\r\n      this.jsonError = '';\r\n    } else {\r\n      // Torna a modalita visuale: prova a parsare il JSON\r\n      this.applyJson();\r\n    }\r\n    this.jsonMode = !this.jsonMode;\r\n  }\r\n\r\n  /**\r\n   * Formatta il JSON nell'editor con indentazione.\r\n   */\r\n  formatJson(): void {\r\n    try {\r\n      const parsed = JSON.parse(this.jsonText);\r\n      this.jsonText = JSON.stringify(parsed, null, 2);\r\n      this.jsonError = '';\r\n    } catch {\r\n      this.jsonError = 'JSON non valido: impossibile formattare.';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambio del testo JSON nell'editor.\r\n   * Tenta il parsing e aggiorna le condizioni se valido.\r\n   * @param text - Testo JSON inserito dall'utente\r\n   */\r\n  onJsonChange(text: string): void {\r\n    this.jsonText = text;\r\n    this.applyJson();\r\n  }\r\n\r\n  /**\r\n   * Aggiunge una nuova condizione con valori predefiniti.\r\n   */\r\n  addCondition(): void {\r\n    const firstField = this.availableFields.length > 0 ? this.availableFields[0].key : '';\r\n    const newCondition: UiFieldCondition = {\r\n      field: firstField,\r\n      operator: 'equals',\r\n      value: '',\r\n    };\r\n    const updated = [...this.conditions, newCondition];\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Rimuove la condizione all'indice specificato.\r\n   * @param index - Indice della condizione da rimuovere\r\n   */\r\n  removeCondition(index: number): void {\r\n    const updated = this.conditions.filter((_, i) => i !== index);\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Rimuove tutte le condizioni.\r\n   */\r\n  clearAll(): void {\r\n    this.emitChange([]);\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambio della logica di combinazione (AND/OR).\r\n   * @param index - Indice della condizione\r\n   * @param logic - Nuova logica ('AND' | 'OR')\r\n   */\r\n  onLogicChange(index: number, logic: 'AND' | 'OR'): void {\r\n    const updated = this.cloneConditions();\r\n    updated[index] = { ...updated[index], logic };\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambio del campo selezionato in una condizione.\r\n   * Resetta il valore quando il campo cambia.\r\n   * @param index - Indice della condizione\r\n   * @param field - Nuova chiave campo\r\n   */\r\n  onFieldChange(index: number, field: string): void {\r\n    const updated = this.cloneConditions();\r\n    updated[index] = { ...updated[index], field, value: '' };\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambio dell'operatore in una condizione.\r\n   * Resetta il valore per operatori senza valore.\r\n   * @param index - Indice della condizione\r\n   * @param operator - Nuovo operatore\r\n   */\r\n  onOperatorChange(index: number, operator: UiConditionalOperator): void {\r\n    const updated = this.cloneConditions();\r\n    const condition = { ...updated[index], operator };\r\n\r\n    // Resetta il valore per operatori che non lo richiedono\r\n    if (this.isNoValueOperator(operator)) {\r\n      condition.value = undefined;\r\n    } else if (this.isArrayOperator(operator) && !Array.isArray(condition.value)) {\r\n      // Converti a array per operatori su array\r\n      condition.value = condition.value ? [condition.value] : [];\r\n    }\r\n\r\n    updated[index] = condition;\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Gestisce il cambio del valore in una condizione.\r\n   * @param index - Indice della condizione\r\n   * @param value - Nuovo valore\r\n   */\r\n  onValueChange(index: number, value: any): void {\r\n    const updated = this.cloneConditions();\r\n    updated[index] = { ...updated[index], value };\r\n    this.emitChange(updated);\r\n  }\r\n\r\n  /**\r\n   * Tenta di applicare il JSON corrente come array di condizioni.\r\n   * Aggiorna l'errore in caso di JSON non valido.\r\n   */\r\n  private applyJson(): void {\r\n    try {\r\n      const parsed = JSON.parse(this.jsonText);\r\n      if (!Array.isArray(parsed)) {\r\n        this.jsonError = 'Il JSON deve essere un array di condizioni.';\r\n        return;\r\n      }\r\n      this.jsonError = '';\r\n      this.emitChange(parsed);\r\n    } catch {\r\n      this.jsonError = 'JSON non valido: controlla la sintassi.';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Clona l'array delle condizioni per garantire immutabilita.\r\n   * @returns Copia superficiale dell'array di condizioni\r\n   */\r\n  private cloneConditions(): UiFieldCondition[] {\r\n    return this.conditions.map((c) => ({ ...c }));\r\n  }\r\n\r\n  /**\r\n   * Emette l'evento di cambio condizioni.\r\n   * @param conditions - Nuova lista di condizioni\r\n   */\r\n  private emitChange(conditions: UiFieldCondition[]): void {\r\n    this.conditions = conditions;\r\n    this.conditionsChange.emit(conditions);\r\n  }\r\n}\r\n"]}