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