@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,611 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter, inject, } from '@angular/core';
|
|
2
|
+
import { FormBuilder, FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
3
|
+
import { MatExpansionModule } from '@angular/material/expansion';
|
|
4
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
5
|
+
import { MatInputModule } from '@angular/material/input';
|
|
6
|
+
import { MatSelectModule } from '@angular/material/select';
|
|
7
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
8
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
9
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
10
|
+
import { Subject } from 'rxjs';
|
|
11
|
+
import { debounceTime, takeUntil } from 'rxjs/operators';
|
|
12
|
+
import { UiValidationEditorComponent } from '../validation-editor/validation-editor.component';
|
|
13
|
+
import { UiOptionsEditorComponent } from '../options-editor/options-editor.component';
|
|
14
|
+
import { UiConditionEditorComponent } from '../condition-editor/condition-editor.component';
|
|
15
|
+
import * as i0 from "@angular/core";
|
|
16
|
+
import * as i1 from "@angular/forms";
|
|
17
|
+
import * as i2 from "@angular/material/expansion";
|
|
18
|
+
import * as i3 from "@angular/material/form-field";
|
|
19
|
+
import * as i4 from "@angular/material/input";
|
|
20
|
+
import * as i5 from "@angular/material/select";
|
|
21
|
+
import * as i6 from "@angular/material/core";
|
|
22
|
+
import * as i7 from "@angular/material/checkbox";
|
|
23
|
+
import * as i8 from "lucide-angular";
|
|
24
|
+
/**
|
|
25
|
+
* Tutti i tipi di campo supportati dal form builder.
|
|
26
|
+
* Utilizzati per popolare la select del tipo nella configurazione campo.
|
|
27
|
+
*/
|
|
28
|
+
const FIELD_TYPES = [
|
|
29
|
+
{ value: 'text', label: 'Testo' },
|
|
30
|
+
{ value: 'email', label: 'Email' },
|
|
31
|
+
{ value: 'password', label: 'Password' },
|
|
32
|
+
{ value: 'number', label: 'Numero' },
|
|
33
|
+
{ value: 'textarea', label: 'Area di testo' },
|
|
34
|
+
{ value: 'select', label: 'Select' },
|
|
35
|
+
{ value: 'multiselect', label: 'Multiselect' },
|
|
36
|
+
{ value: 'freemultiselect', label: 'Multiselect libero' },
|
|
37
|
+
{ value: 'checkbox', label: 'Checkbox' },
|
|
38
|
+
{ value: 'switch', label: 'Switch' },
|
|
39
|
+
{ value: 'radio', label: 'Radio' },
|
|
40
|
+
{ value: 'date', label: 'Data' },
|
|
41
|
+
{ value: 'datetime', label: 'Data e ora' },
|
|
42
|
+
{ value: 'file', label: 'File' },
|
|
43
|
+
{ value: 'custom', label: 'Custom' },
|
|
44
|
+
{ value: 'divider', label: 'Separatore' },
|
|
45
|
+
{ value: 'flag', label: 'Flag' },
|
|
46
|
+
{ value: 'location', label: 'Location' },
|
|
47
|
+
{ value: 'location-table', label: 'Location Table' },
|
|
48
|
+
];
|
|
49
|
+
/**
|
|
50
|
+
* Tipi di campo che supportano la configurazione delle opzioni.
|
|
51
|
+
* Per questi tipi viene mostrato il pannello opzioni nell'editor.
|
|
52
|
+
*/
|
|
53
|
+
const OPTIONS_FIELD_TYPES = ['select', 'multiselect', 'freemultiselect', 'radio'];
|
|
54
|
+
/**
|
|
55
|
+
* Pannello laterale destro per la configurazione di un campo selezionato.
|
|
56
|
+
*
|
|
57
|
+
* Mostra un form reattivo suddiviso in pannelli espandibili per gestire
|
|
58
|
+
* proprieta base, stato, layout, validazioni, opzioni e condizioni.
|
|
59
|
+
* Emette aggiornamenti parziali con debounce di 300ms.
|
|
60
|
+
*
|
|
61
|
+
* @selector ui-editor-field-config-panel
|
|
62
|
+
*/
|
|
63
|
+
export class UiEditorFieldConfigPanelComponent {
|
|
64
|
+
constructor() {
|
|
65
|
+
/** @internal Servizio per la costruzione del FormGroup. */
|
|
66
|
+
this.fb = inject(FormBuilder);
|
|
67
|
+
/** @internal Subject per la distruzione e pulizia delle sottoscrizioni. */
|
|
68
|
+
this.destroy$ = new Subject();
|
|
69
|
+
/** Campo attualmente selezionato per la configurazione. */
|
|
70
|
+
this.field = null;
|
|
71
|
+
/** Lista dei campi disponibili per le condizioni. */
|
|
72
|
+
this.availableFields = [];
|
|
73
|
+
/** Emesso quando il campo viene aggiornato dall'utente. */
|
|
74
|
+
this.fieldUpdate = new EventEmitter();
|
|
75
|
+
/** Form reattivo per la configurazione del campo. */
|
|
76
|
+
this.configForm = null;
|
|
77
|
+
/** Lista dei tipi di campo per la select. */
|
|
78
|
+
this.fieldTypes = FIELD_TYPES;
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Indica se il campo selezionato supporta le opzioni.
|
|
82
|
+
* Viene utilizzato per mostrare/nascondere il pannello opzioni.
|
|
83
|
+
*/
|
|
84
|
+
get isOptionsField() {
|
|
85
|
+
if (!this.field)
|
|
86
|
+
return false;
|
|
87
|
+
return OPTIONS_FIELD_TYPES.includes(this.field.type);
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Rileva cambiamenti sugli input e ricostruisce il form
|
|
91
|
+
* quando cambia il campo selezionato.
|
|
92
|
+
*/
|
|
93
|
+
ngOnChanges(changes) {
|
|
94
|
+
if (changes['field']) {
|
|
95
|
+
this.buildForm();
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
ngOnDestroy() {
|
|
99
|
+
this.destroy$.next();
|
|
100
|
+
this.destroy$.complete();
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Restituisce le opzioni del campo come array statico.
|
|
104
|
+
* Ignora opzioni osservabili poiche non gestibili nell'editor.
|
|
105
|
+
*/
|
|
106
|
+
getFieldOptions() {
|
|
107
|
+
if (!this.field?.options || !Array.isArray(this.field.options))
|
|
108
|
+
return [];
|
|
109
|
+
return this.field.options.map((o) => ({ value: o.value, label: o.label }));
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Gestisce l'aggiornamento delle validazioni dal sotto-componente.
|
|
113
|
+
* @param validations - Nuova lista di regole di validazione
|
|
114
|
+
*/
|
|
115
|
+
onValidationsChange(validations) {
|
|
116
|
+
this.fieldUpdate.emit({ validation: validations });
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Gestisce l'aggiornamento delle opzioni dal sotto-componente.
|
|
120
|
+
* @param options - Nuova lista di opzioni
|
|
121
|
+
*/
|
|
122
|
+
onOptionsChange(options) {
|
|
123
|
+
this.fieldUpdate.emit({ options });
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Gestisce l'aggiornamento delle condizioni di visibilita.
|
|
127
|
+
* @param conditions - Nuova lista di condizioni
|
|
128
|
+
*/
|
|
129
|
+
onVisibilityConditionsChange(conditions) {
|
|
130
|
+
this.fieldUpdate.emit({ conditions });
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Gestisce l'aggiornamento delle condizioni di disabilitazione.
|
|
134
|
+
* @param conditions - Nuova lista di condizioni
|
|
135
|
+
*/
|
|
136
|
+
onDisableConditionsChange(conditions) {
|
|
137
|
+
this.fieldUpdate.emit({ disableConditions: conditions });
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* @internal
|
|
141
|
+
* Costruisce il FormGroup a partire dal campo selezionato.
|
|
142
|
+
* Si sottoscrive ai valueChanges con debounce per emettere aggiornamenti.
|
|
143
|
+
*/
|
|
144
|
+
buildForm() {
|
|
145
|
+
this.destroy$.next();
|
|
146
|
+
if (!this.field) {
|
|
147
|
+
this.configForm = null;
|
|
148
|
+
return;
|
|
149
|
+
}
|
|
150
|
+
this.configForm = this.fb.group({
|
|
151
|
+
key: [this.field.key],
|
|
152
|
+
type: [this.field.type],
|
|
153
|
+
label: [this.field.label],
|
|
154
|
+
placeholder: [this.field.placeholder || ''],
|
|
155
|
+
tooltip: [this.field.tooltip || ''],
|
|
156
|
+
defaultValue: [this.field.defaultValue ?? ''],
|
|
157
|
+
required: [this.field.required || false],
|
|
158
|
+
disabled: [this.field.disabled || false],
|
|
159
|
+
readonly: [this.field.readonly || false],
|
|
160
|
+
hideEmptyOption: [this.field.hideEmptyOption || false],
|
|
161
|
+
columns: [this.field.layout?.columns ?? null],
|
|
162
|
+
order: [this.field.layout?.order ?? null],
|
|
163
|
+
});
|
|
164
|
+
this.configForm.valueChanges.pipe(debounceTime(300), takeUntil(this.destroy$)).subscribe((values) => {
|
|
165
|
+
/** Mappa i valori del form nel formato di aggiornamento parziale del campo */
|
|
166
|
+
const update = {
|
|
167
|
+
key: values.key,
|
|
168
|
+
type: values.type,
|
|
169
|
+
label: values.label,
|
|
170
|
+
placeholder: values.placeholder || undefined,
|
|
171
|
+
tooltip: values.tooltip || undefined,
|
|
172
|
+
defaultValue: values.defaultValue || undefined,
|
|
173
|
+
required: values.required,
|
|
174
|
+
disabled: values.disabled,
|
|
175
|
+
readonly: values.readonly,
|
|
176
|
+
hideEmptyOption: values.hideEmptyOption,
|
|
177
|
+
layout: {
|
|
178
|
+
columns: values.columns ?? undefined,
|
|
179
|
+
order: values.order ?? undefined,
|
|
180
|
+
},
|
|
181
|
+
};
|
|
182
|
+
this.fieldUpdate.emit(update);
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFieldConfigPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
186
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiEditorFieldConfigPanelComponent, isStandalone: true, selector: "ui-editor-field-config-panel", inputs: { field: "field", availableFields: "availableFields" }, outputs: { fieldUpdate: "fieldUpdate" }, usesOnChanges: true, ngImport: i0, template: `
|
|
187
|
+
<div class="field-config-panel">
|
|
188
|
+
<!-- Intestazione del pannello -->
|
|
189
|
+
<div class="field-config-panel__header">
|
|
190
|
+
<lucide-icon name="settings" [size]="18" aria-hidden="true" />
|
|
191
|
+
<span class="field-config-panel__title">Configurazione Campo</span>
|
|
192
|
+
</div>
|
|
193
|
+
|
|
194
|
+
<!-- Stato vuoto: nessun campo selezionato -->
|
|
195
|
+
@if (!field) {
|
|
196
|
+
<div class="field-config-panel__empty">
|
|
197
|
+
<lucide-icon name="mouse-pointer-click" [size]="32" aria-hidden="true" />
|
|
198
|
+
<p class="field-config-panel__empty-text">
|
|
199
|
+
Seleziona un campo per configurarlo
|
|
200
|
+
</p>
|
|
201
|
+
</div>
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
<!-- Form di configurazione del campo -->
|
|
205
|
+
@if (field && configForm) {
|
|
206
|
+
<div class="field-config-panel__content">
|
|
207
|
+
<mat-accordion multi>
|
|
208
|
+
|
|
209
|
+
<!-- Pannello 1: Proprieta base -->
|
|
210
|
+
<mat-expansion-panel [expanded]="true">
|
|
211
|
+
<mat-expansion-panel-header>
|
|
212
|
+
<mat-panel-title>
|
|
213
|
+
<lucide-icon name="file-text" [size]="16" aria-hidden="true" />
|
|
214
|
+
<span>Proprieta base</span>
|
|
215
|
+
</mat-panel-title>
|
|
216
|
+
</mat-expansion-panel-header>
|
|
217
|
+
|
|
218
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
219
|
+
<!-- Chiave univoca -->
|
|
220
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
221
|
+
<mat-label>Chiave (key)</mat-label>
|
|
222
|
+
<input matInput formControlName="key" placeholder="campo_univoco" />
|
|
223
|
+
<mat-hint>Identificativo univoco del campo</mat-hint>
|
|
224
|
+
</mat-form-field>
|
|
225
|
+
|
|
226
|
+
<!-- Tipo di campo -->
|
|
227
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
228
|
+
<mat-label>Tipo</mat-label>
|
|
229
|
+
<mat-select formControlName="type">
|
|
230
|
+
@for (ft of fieldTypes; track ft.value) {
|
|
231
|
+
<mat-option [value]="ft.value">{{ ft.label }}</mat-option>
|
|
232
|
+
}
|
|
233
|
+
</mat-select>
|
|
234
|
+
</mat-form-field>
|
|
235
|
+
|
|
236
|
+
<!-- Label visualizzata -->
|
|
237
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
238
|
+
<mat-label>Label</mat-label>
|
|
239
|
+
<input matInput formControlName="label" placeholder="Etichetta del campo" />
|
|
240
|
+
</mat-form-field>
|
|
241
|
+
|
|
242
|
+
<!-- Placeholder -->
|
|
243
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
244
|
+
<mat-label>Placeholder</mat-label>
|
|
245
|
+
<input matInput formControlName="placeholder" placeholder="Testo segnaposto" />
|
|
246
|
+
</mat-form-field>
|
|
247
|
+
|
|
248
|
+
<!-- Tooltip -->
|
|
249
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
250
|
+
<mat-label>Tooltip</mat-label>
|
|
251
|
+
<input matInput formControlName="tooltip" placeholder="Testo di aiuto" />
|
|
252
|
+
</mat-form-field>
|
|
253
|
+
|
|
254
|
+
<!-- Valore di default -->
|
|
255
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
256
|
+
<mat-label>Valore di default</mat-label>
|
|
257
|
+
<input matInput formControlName="defaultValue" placeholder="Valore iniziale" />
|
|
258
|
+
</mat-form-field>
|
|
259
|
+
</div>
|
|
260
|
+
</mat-expansion-panel>
|
|
261
|
+
|
|
262
|
+
<!-- Pannello 2: Stato del campo -->
|
|
263
|
+
<mat-expansion-panel>
|
|
264
|
+
<mat-expansion-panel-header>
|
|
265
|
+
<mat-panel-title>
|
|
266
|
+
<lucide-icon name="toggle-left" [size]="16" aria-hidden="true" />
|
|
267
|
+
<span>Stato campo</span>
|
|
268
|
+
</mat-panel-title>
|
|
269
|
+
</mat-expansion-panel-header>
|
|
270
|
+
|
|
271
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
272
|
+
<mat-checkbox formControlName="required" color="primary">
|
|
273
|
+
Obbligatorio
|
|
274
|
+
</mat-checkbox>
|
|
275
|
+
|
|
276
|
+
<mat-checkbox formControlName="disabled" color="primary">
|
|
277
|
+
Disabilitato
|
|
278
|
+
</mat-checkbox>
|
|
279
|
+
|
|
280
|
+
<mat-checkbox formControlName="readonly" color="primary">
|
|
281
|
+
Sola lettura
|
|
282
|
+
</mat-checkbox>
|
|
283
|
+
|
|
284
|
+
@if (isOptionsField) {
|
|
285
|
+
<mat-checkbox formControlName="hideEmptyOption" color="primary">
|
|
286
|
+
Nascondi opzione vuota
|
|
287
|
+
</mat-checkbox>
|
|
288
|
+
}
|
|
289
|
+
</div>
|
|
290
|
+
</mat-expansion-panel>
|
|
291
|
+
|
|
292
|
+
<!-- Pannello 3: Layout -->
|
|
293
|
+
<mat-expansion-panel>
|
|
294
|
+
<mat-expansion-panel-header>
|
|
295
|
+
<mat-panel-title>
|
|
296
|
+
<lucide-icon name="layout-grid" [size]="16" aria-hidden="true" />
|
|
297
|
+
<span>Layout</span>
|
|
298
|
+
</mat-panel-title>
|
|
299
|
+
</mat-expansion-panel-header>
|
|
300
|
+
|
|
301
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
302
|
+
<!-- Colonne -->
|
|
303
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
304
|
+
<mat-label>Colonne (1-12)</mat-label>
|
|
305
|
+
<input matInput type="number" formControlName="columns" min="1" max="12" />
|
|
306
|
+
<mat-hint>Numero di colonne nel grid</mat-hint>
|
|
307
|
+
</mat-form-field>
|
|
308
|
+
|
|
309
|
+
<!-- Ordine -->
|
|
310
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
311
|
+
<mat-label>Ordine</mat-label>
|
|
312
|
+
<input matInput type="number" formControlName="order" />
|
|
313
|
+
<mat-hint>Posizione nel grid</mat-hint>
|
|
314
|
+
</mat-form-field>
|
|
315
|
+
</div>
|
|
316
|
+
</mat-expansion-panel>
|
|
317
|
+
|
|
318
|
+
<!-- Pannello 4: Validazioni -->
|
|
319
|
+
<mat-expansion-panel>
|
|
320
|
+
<mat-expansion-panel-header>
|
|
321
|
+
<mat-panel-title>
|
|
322
|
+
<lucide-icon name="shield-check" [size]="16" aria-hidden="true" />
|
|
323
|
+
<span>Validazioni</span>
|
|
324
|
+
</mat-panel-title>
|
|
325
|
+
</mat-expansion-panel-header>
|
|
326
|
+
|
|
327
|
+
<ui-validation-editor
|
|
328
|
+
[validations]="field.validation || []"
|
|
329
|
+
(validationsChange)="onValidationsChange($event)"
|
|
330
|
+
/>
|
|
331
|
+
</mat-expansion-panel>
|
|
332
|
+
|
|
333
|
+
<!-- Pannello 5: Opzioni (solo per campi con opzioni) -->
|
|
334
|
+
@if (isOptionsField) {
|
|
335
|
+
<mat-expansion-panel>
|
|
336
|
+
<mat-expansion-panel-header>
|
|
337
|
+
<mat-panel-title>
|
|
338
|
+
<lucide-icon name="list" [size]="16" aria-hidden="true" />
|
|
339
|
+
<span>Opzioni</span>
|
|
340
|
+
</mat-panel-title>
|
|
341
|
+
</mat-expansion-panel-header>
|
|
342
|
+
|
|
343
|
+
<ui-options-editor
|
|
344
|
+
[options]="getFieldOptions()"
|
|
345
|
+
(optionsChange)="onOptionsChange($event)"
|
|
346
|
+
/>
|
|
347
|
+
</mat-expansion-panel>
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
<!-- Pannello 6: Condizioni di visibilita -->
|
|
351
|
+
<mat-expansion-panel>
|
|
352
|
+
<mat-expansion-panel-header>
|
|
353
|
+
<mat-panel-title>
|
|
354
|
+
<lucide-icon name="eye" [size]="16" aria-hidden="true" />
|
|
355
|
+
<span>Condizioni di visibilita</span>
|
|
356
|
+
</mat-panel-title>
|
|
357
|
+
</mat-expansion-panel-header>
|
|
358
|
+
|
|
359
|
+
<ui-condition-editor
|
|
360
|
+
[conditions]="field.conditions || []"
|
|
361
|
+
[availableFields]="availableFields"
|
|
362
|
+
(conditionsChange)="onVisibilityConditionsChange($event)"
|
|
363
|
+
/>
|
|
364
|
+
</mat-expansion-panel>
|
|
365
|
+
|
|
366
|
+
<!-- Pannello 7: Condizioni di disabilitazione -->
|
|
367
|
+
<mat-expansion-panel>
|
|
368
|
+
<mat-expansion-panel-header>
|
|
369
|
+
<mat-panel-title>
|
|
370
|
+
<lucide-icon name="ban" [size]="16" aria-hidden="true" />
|
|
371
|
+
<span>Condizioni di disabilitazione</span>
|
|
372
|
+
</mat-panel-title>
|
|
373
|
+
</mat-expansion-panel-header>
|
|
374
|
+
|
|
375
|
+
<ui-condition-editor
|
|
376
|
+
[conditions]="field.disableConditions || []"
|
|
377
|
+
[availableFields]="availableFields"
|
|
378
|
+
(conditionsChange)="onDisableConditionsChange($event)"
|
|
379
|
+
/>
|
|
380
|
+
</mat-expansion-panel>
|
|
381
|
+
|
|
382
|
+
</mat-accordion>
|
|
383
|
+
</div>
|
|
384
|
+
}
|
|
385
|
+
</div>
|
|
386
|
+
`, isInline: true, styles: [".field-config-panel{display:flex;flex-direction:column;height:100%;background:var(--ui-color-surface, #fff);border-left:1px solid var(--ui-color-border, #e0e0e0);overflow-y:auto}.field-config-panel__header{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-3, 12px) var(--ui-spacing-4, 16px);border-bottom:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-bg-subtle, #fafafa)}.field-config-panel__title{font-weight:600;font-size:var(--ui-font-size-sm, .875rem);color:var(--ui-color-text, #212121)}.field-config-panel__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-3, 12px);padding:var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);text-align:center;color:var(--ui-color-text-muted, #9e9e9e)}.field-config-panel__empty-text{font-size:var(--ui-font-size-sm, .875rem);margin:0}.field-config-panel__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-2, 8px)}.field-config-panel__fields{display:flex;flex-direction:column;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-1, 4px) 0}.field-config-panel__field{width:100%}mat-panel-title{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);font-size:var(--ui-font-size-sm, .875rem);font-weight:500;color:var(--ui-color-text, #212121)}mat-checkbox{margin:var(--ui-spacing-1, 4px) 0}\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.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i1.MaxValidator, selector: "input[type=number][max][formControlName],input[type=number][max][formControl],input[type=number][max][ngModel]", inputs: ["max"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "directive", type: i2.MatAccordion, selector: "mat-accordion", inputs: ["hideToggle", "displayMode", "togglePosition"], exportAs: ["matAccordion"] }, { kind: "component", type: i2.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i2.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i2.MatExpansionPanelTitle, selector: "mat-panel-title" }, { 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: "directive", type: i3.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { 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: MatCheckboxModule }, { kind: "component", type: i7.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i8.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiValidationEditorComponent, selector: "ui-validation-editor", inputs: ["validations"], outputs: ["validationsChange"] }, { kind: "component", type: UiOptionsEditorComponent, selector: "ui-options-editor", inputs: ["options"], outputs: ["optionsChange"] }, { kind: "component", type: UiConditionEditorComponent, selector: "ui-condition-editor", inputs: ["conditions", "title", "availableFields"], outputs: ["conditionsChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
387
|
+
}
|
|
388
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFieldConfigPanelComponent, decorators: [{
|
|
389
|
+
type: Component,
|
|
390
|
+
args: [{ selector: 'ui-editor-field-config-panel', standalone: true, imports: [
|
|
391
|
+
FormsModule,
|
|
392
|
+
ReactiveFormsModule,
|
|
393
|
+
MatExpansionModule,
|
|
394
|
+
MatFormFieldModule,
|
|
395
|
+
MatInputModule,
|
|
396
|
+
MatSelectModule,
|
|
397
|
+
MatCheckboxModule,
|
|
398
|
+
MatTooltipModule,
|
|
399
|
+
LucideAngularModule,
|
|
400
|
+
UiValidationEditorComponent,
|
|
401
|
+
UiOptionsEditorComponent,
|
|
402
|
+
UiConditionEditorComponent,
|
|
403
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
404
|
+
<div class="field-config-panel">
|
|
405
|
+
<!-- Intestazione del pannello -->
|
|
406
|
+
<div class="field-config-panel__header">
|
|
407
|
+
<lucide-icon name="settings" [size]="18" aria-hidden="true" />
|
|
408
|
+
<span class="field-config-panel__title">Configurazione Campo</span>
|
|
409
|
+
</div>
|
|
410
|
+
|
|
411
|
+
<!-- Stato vuoto: nessun campo selezionato -->
|
|
412
|
+
@if (!field) {
|
|
413
|
+
<div class="field-config-panel__empty">
|
|
414
|
+
<lucide-icon name="mouse-pointer-click" [size]="32" aria-hidden="true" />
|
|
415
|
+
<p class="field-config-panel__empty-text">
|
|
416
|
+
Seleziona un campo per configurarlo
|
|
417
|
+
</p>
|
|
418
|
+
</div>
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
<!-- Form di configurazione del campo -->
|
|
422
|
+
@if (field && configForm) {
|
|
423
|
+
<div class="field-config-panel__content">
|
|
424
|
+
<mat-accordion multi>
|
|
425
|
+
|
|
426
|
+
<!-- Pannello 1: Proprieta base -->
|
|
427
|
+
<mat-expansion-panel [expanded]="true">
|
|
428
|
+
<mat-expansion-panel-header>
|
|
429
|
+
<mat-panel-title>
|
|
430
|
+
<lucide-icon name="file-text" [size]="16" aria-hidden="true" />
|
|
431
|
+
<span>Proprieta base</span>
|
|
432
|
+
</mat-panel-title>
|
|
433
|
+
</mat-expansion-panel-header>
|
|
434
|
+
|
|
435
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
436
|
+
<!-- Chiave univoca -->
|
|
437
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
438
|
+
<mat-label>Chiave (key)</mat-label>
|
|
439
|
+
<input matInput formControlName="key" placeholder="campo_univoco" />
|
|
440
|
+
<mat-hint>Identificativo univoco del campo</mat-hint>
|
|
441
|
+
</mat-form-field>
|
|
442
|
+
|
|
443
|
+
<!-- Tipo di campo -->
|
|
444
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
445
|
+
<mat-label>Tipo</mat-label>
|
|
446
|
+
<mat-select formControlName="type">
|
|
447
|
+
@for (ft of fieldTypes; track ft.value) {
|
|
448
|
+
<mat-option [value]="ft.value">{{ ft.label }}</mat-option>
|
|
449
|
+
}
|
|
450
|
+
</mat-select>
|
|
451
|
+
</mat-form-field>
|
|
452
|
+
|
|
453
|
+
<!-- Label visualizzata -->
|
|
454
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
455
|
+
<mat-label>Label</mat-label>
|
|
456
|
+
<input matInput formControlName="label" placeholder="Etichetta del campo" />
|
|
457
|
+
</mat-form-field>
|
|
458
|
+
|
|
459
|
+
<!-- Placeholder -->
|
|
460
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
461
|
+
<mat-label>Placeholder</mat-label>
|
|
462
|
+
<input matInput formControlName="placeholder" placeholder="Testo segnaposto" />
|
|
463
|
+
</mat-form-field>
|
|
464
|
+
|
|
465
|
+
<!-- Tooltip -->
|
|
466
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
467
|
+
<mat-label>Tooltip</mat-label>
|
|
468
|
+
<input matInput formControlName="tooltip" placeholder="Testo di aiuto" />
|
|
469
|
+
</mat-form-field>
|
|
470
|
+
|
|
471
|
+
<!-- Valore di default -->
|
|
472
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
473
|
+
<mat-label>Valore di default</mat-label>
|
|
474
|
+
<input matInput formControlName="defaultValue" placeholder="Valore iniziale" />
|
|
475
|
+
</mat-form-field>
|
|
476
|
+
</div>
|
|
477
|
+
</mat-expansion-panel>
|
|
478
|
+
|
|
479
|
+
<!-- Pannello 2: Stato del campo -->
|
|
480
|
+
<mat-expansion-panel>
|
|
481
|
+
<mat-expansion-panel-header>
|
|
482
|
+
<mat-panel-title>
|
|
483
|
+
<lucide-icon name="toggle-left" [size]="16" aria-hidden="true" />
|
|
484
|
+
<span>Stato campo</span>
|
|
485
|
+
</mat-panel-title>
|
|
486
|
+
</mat-expansion-panel-header>
|
|
487
|
+
|
|
488
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
489
|
+
<mat-checkbox formControlName="required" color="primary">
|
|
490
|
+
Obbligatorio
|
|
491
|
+
</mat-checkbox>
|
|
492
|
+
|
|
493
|
+
<mat-checkbox formControlName="disabled" color="primary">
|
|
494
|
+
Disabilitato
|
|
495
|
+
</mat-checkbox>
|
|
496
|
+
|
|
497
|
+
<mat-checkbox formControlName="readonly" color="primary">
|
|
498
|
+
Sola lettura
|
|
499
|
+
</mat-checkbox>
|
|
500
|
+
|
|
501
|
+
@if (isOptionsField) {
|
|
502
|
+
<mat-checkbox formControlName="hideEmptyOption" color="primary">
|
|
503
|
+
Nascondi opzione vuota
|
|
504
|
+
</mat-checkbox>
|
|
505
|
+
}
|
|
506
|
+
</div>
|
|
507
|
+
</mat-expansion-panel>
|
|
508
|
+
|
|
509
|
+
<!-- Pannello 3: Layout -->
|
|
510
|
+
<mat-expansion-panel>
|
|
511
|
+
<mat-expansion-panel-header>
|
|
512
|
+
<mat-panel-title>
|
|
513
|
+
<lucide-icon name="layout-grid" [size]="16" aria-hidden="true" />
|
|
514
|
+
<span>Layout</span>
|
|
515
|
+
</mat-panel-title>
|
|
516
|
+
</mat-expansion-panel-header>
|
|
517
|
+
|
|
518
|
+
<div class="field-config-panel__fields" [formGroup]="configForm">
|
|
519
|
+
<!-- Colonne -->
|
|
520
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
521
|
+
<mat-label>Colonne (1-12)</mat-label>
|
|
522
|
+
<input matInput type="number" formControlName="columns" min="1" max="12" />
|
|
523
|
+
<mat-hint>Numero di colonne nel grid</mat-hint>
|
|
524
|
+
</mat-form-field>
|
|
525
|
+
|
|
526
|
+
<!-- Ordine -->
|
|
527
|
+
<mat-form-field appearance="outline" class="field-config-panel__field">
|
|
528
|
+
<mat-label>Ordine</mat-label>
|
|
529
|
+
<input matInput type="number" formControlName="order" />
|
|
530
|
+
<mat-hint>Posizione nel grid</mat-hint>
|
|
531
|
+
</mat-form-field>
|
|
532
|
+
</div>
|
|
533
|
+
</mat-expansion-panel>
|
|
534
|
+
|
|
535
|
+
<!-- Pannello 4: Validazioni -->
|
|
536
|
+
<mat-expansion-panel>
|
|
537
|
+
<mat-expansion-panel-header>
|
|
538
|
+
<mat-panel-title>
|
|
539
|
+
<lucide-icon name="shield-check" [size]="16" aria-hidden="true" />
|
|
540
|
+
<span>Validazioni</span>
|
|
541
|
+
</mat-panel-title>
|
|
542
|
+
</mat-expansion-panel-header>
|
|
543
|
+
|
|
544
|
+
<ui-validation-editor
|
|
545
|
+
[validations]="field.validation || []"
|
|
546
|
+
(validationsChange)="onValidationsChange($event)"
|
|
547
|
+
/>
|
|
548
|
+
</mat-expansion-panel>
|
|
549
|
+
|
|
550
|
+
<!-- Pannello 5: Opzioni (solo per campi con opzioni) -->
|
|
551
|
+
@if (isOptionsField) {
|
|
552
|
+
<mat-expansion-panel>
|
|
553
|
+
<mat-expansion-panel-header>
|
|
554
|
+
<mat-panel-title>
|
|
555
|
+
<lucide-icon name="list" [size]="16" aria-hidden="true" />
|
|
556
|
+
<span>Opzioni</span>
|
|
557
|
+
</mat-panel-title>
|
|
558
|
+
</mat-expansion-panel-header>
|
|
559
|
+
|
|
560
|
+
<ui-options-editor
|
|
561
|
+
[options]="getFieldOptions()"
|
|
562
|
+
(optionsChange)="onOptionsChange($event)"
|
|
563
|
+
/>
|
|
564
|
+
</mat-expansion-panel>
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
<!-- Pannello 6: Condizioni di visibilita -->
|
|
568
|
+
<mat-expansion-panel>
|
|
569
|
+
<mat-expansion-panel-header>
|
|
570
|
+
<mat-panel-title>
|
|
571
|
+
<lucide-icon name="eye" [size]="16" aria-hidden="true" />
|
|
572
|
+
<span>Condizioni di visibilita</span>
|
|
573
|
+
</mat-panel-title>
|
|
574
|
+
</mat-expansion-panel-header>
|
|
575
|
+
|
|
576
|
+
<ui-condition-editor
|
|
577
|
+
[conditions]="field.conditions || []"
|
|
578
|
+
[availableFields]="availableFields"
|
|
579
|
+
(conditionsChange)="onVisibilityConditionsChange($event)"
|
|
580
|
+
/>
|
|
581
|
+
</mat-expansion-panel>
|
|
582
|
+
|
|
583
|
+
<!-- Pannello 7: Condizioni di disabilitazione -->
|
|
584
|
+
<mat-expansion-panel>
|
|
585
|
+
<mat-expansion-panel-header>
|
|
586
|
+
<mat-panel-title>
|
|
587
|
+
<lucide-icon name="ban" [size]="16" aria-hidden="true" />
|
|
588
|
+
<span>Condizioni di disabilitazione</span>
|
|
589
|
+
</mat-panel-title>
|
|
590
|
+
</mat-expansion-panel-header>
|
|
591
|
+
|
|
592
|
+
<ui-condition-editor
|
|
593
|
+
[conditions]="field.disableConditions || []"
|
|
594
|
+
[availableFields]="availableFields"
|
|
595
|
+
(conditionsChange)="onDisableConditionsChange($event)"
|
|
596
|
+
/>
|
|
597
|
+
</mat-expansion-panel>
|
|
598
|
+
|
|
599
|
+
</mat-accordion>
|
|
600
|
+
</div>
|
|
601
|
+
}
|
|
602
|
+
</div>
|
|
603
|
+
`, styles: [".field-config-panel{display:flex;flex-direction:column;height:100%;background:var(--ui-color-surface, #fff);border-left:1px solid var(--ui-color-border, #e0e0e0);overflow-y:auto}.field-config-panel__header{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-3, 12px) var(--ui-spacing-4, 16px);border-bottom:1px solid var(--ui-color-border, #e0e0e0);background:var(--ui-color-bg-subtle, #fafafa)}.field-config-panel__title{font-weight:600;font-size:var(--ui-font-size-sm, .875rem);color:var(--ui-color-text, #212121)}.field-config-panel__empty{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-3, 12px);padding:var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);text-align:center;color:var(--ui-color-text-muted, #9e9e9e)}.field-config-panel__empty-text{font-size:var(--ui-font-size-sm, .875rem);margin:0}.field-config-panel__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-2, 8px)}.field-config-panel__fields{display:flex;flex-direction:column;gap:var(--ui-spacing-2, 8px);padding:var(--ui-spacing-1, 4px) 0}.field-config-panel__field{width:100%}mat-panel-title{display:flex;align-items:center;gap:var(--ui-spacing-2, 8px);font-size:var(--ui-font-size-sm, .875rem);font-weight:500;color:var(--ui-color-text, #212121)}mat-checkbox{margin:var(--ui-spacing-1, 4px) 0}\n"] }]
|
|
604
|
+
}], propDecorators: { field: [{
|
|
605
|
+
type: Input
|
|
606
|
+
}], availableFields: [{
|
|
607
|
+
type: Input
|
|
608
|
+
}], fieldUpdate: [{
|
|
609
|
+
type: Output
|
|
610
|
+
}] } });
|
|
611
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"field-config-panel.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,uBAAuB,EACvB,KAAK,EACL,MAAM,EACN,YAAY,EAIZ,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,WAAW,EAAa,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC1F,OAAO,EAAE,kBAAkB,EAAE,MAAM,6BAA6B,CAAC;AACjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,4BAA4B,CAAC;AAC/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAIzD,OAAO,EAAE,2BAA2B,EAAE,MAAM,kDAAkD,CAAC;AAC/F,OAAO,EAAE,wBAAwB,EAAE,MAAM,4CAA4C,CAAC;AACtF,OAAO,EAAE,0BAA0B,EAAE,MAAM,gDAAgD,CAAC;;;;;;;;;;AAE5F;;;GAGG;AACH,MAAM,WAAW,GAA4C;IAC3D,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE;IACjC,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE;IAClC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;IACxC,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;IACpC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,eAAe,EAAE;IAC7C,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;IACpC,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,aAAa,EAAE;IAC9C,EAAE,KAAK,EAAE,iBAAiB,EAAE,KAAK,EAAE,oBAAoB,EAAE;IACzD,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;IACxC,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;IACpC,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE;IAClC,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;IAChC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,YAAY,EAAE;IAC1C,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;IAChC,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;IACpC,EAAE,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,YAAY,EAAE;IACzC,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;IAChC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE;IACxC,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,gBAAgB,EAAE;CACrD,CAAC;AAEF;;;GAGG;AACH,MAAM,mBAAmB,GAAkB,CAAC,QAAQ,EAAE,aAAa,EAAE,iBAAiB,EAAE,OAAO,CAAC,CAAC;AAEjG;;;;;;;;GAQG;AA8SH,MAAM,OAAO,iCAAiC;IA7S9C;QA8SE,2DAA2D;QAC1C,OAAE,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAE1C,2EAA2E;QAC1D,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAEhD,2DAA2D;QAClD,UAAK,GAAiC,IAAI,CAAC;QAEpD,qDAAqD;QAC5C,oBAAe,GAA6B,EAAE,CAAC;QAExD,2DAA2D;QACjD,gBAAW,GAAG,IAAI,YAAY,EAAkC,CAAC;QAE3E,qDAAqD;QACrD,eAAU,GAAqB,IAAI,CAAC;QAEpC,6CAA6C;QACpC,eAAU,GAAG,WAAW,CAAC;KAoHnC;IAlHC;;;OAGG;IACH,IAAI,cAAc;QAChB,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,OAAO,KAAK,CAAC;QAC9B,OAAO,mBAAmB,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACvD,CAAC;IAED;;;OAGG;IACH,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YACrB,IAAI,CAAC,SAAS,EAAE,CAAC;QACnB,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC;YAAE,OAAO,EAAE,CAAC;QAC1E,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IAC7E,CAAC;IAED;;;OAGG;IACH,mBAAmB,CAAC,WAAkB;QACpC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,UAAU,EAAE,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,OAAwC;QACtD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;IACrC,CAAC;IAED;;;OAGG;IACH,4BAA4B,CAAC,UAAiB;QAC5C,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC;IACxC,CAAC;IAED;;;OAGG;IACH,yBAAyB,CAAC,UAAiB;QACzC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,iBAAiB,EAAE,UAAU,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;;;OAIG;IACK,SAAS;QACf,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAErB,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAChB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC;YAC9B,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC;YACrB,IAAI,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;YACvB,KAAK,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC;YACzB,WAAW,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,EAAE,CAAC;YAC3C,OAAO,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,EAAE,CAAC;YACnC,YAAY,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,EAAE,CAAC;YAC7C,QAAQ,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;YACxC,QAAQ,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;YACxC,QAAQ,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;YACxC,eAAe,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,IAAI,KAAK,CAAC;YACtD,OAAO,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,OAAO,IAAI,IAAI,CAAC;YAC7C,KAAK,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC;SAC1C,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;YAClG,8EAA8E;YAC9E,MAAM,MAAM,GAAmC;gBAC7C,GAAG,EAAE,MAAM,CAAC,GAAG;gBACf,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,KAAK,EAAE,MAAM,CAAC,KAAK;gBACnB,WAAW,EAAE,MAAM,CAAC,WAAW,IAAI,SAAS;gBAC5C,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,SAAS;gBACpC,YAAY,EAAE,MAAM,CAAC,YAAY,IAAI,SAAS;gBAC9C,QAAQ,EAAE,MAAM,CAAC,QAAQ;gBACzB,QAAQ,EAAE,MAAM,CAAC,QAAQ;gBACzB,QAAQ,EAAE,MAAM,CAAC,QAAQ;gBACzB,eAAe,EAAE,MAAM,CAAC,eAAe;gBACvC,MAAM,EAAE;oBACN,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,SAAS;oBACpC,KAAK,EAAE,MAAM,CAAC,KAAK,IAAI,SAAS;iBACjC;aACF,CAAC;YACF,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;IACL,CAAC;+GAvIU,iCAAiC;mGAAjC,iCAAiC,sNA3RlC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwMT,s5CAtNC,WAAW,0kCACX,mBAAmB,+UACnB,kBAAkB,unBAClB,kBAAkB,kYAClB,cAAc,0WACd,eAAe,mrBACf,iBAAiB,oYACjB,gBAAgB,8BAChB,mBAAmB,gPACnB,2BAA2B,0HAC3B,wBAAwB,+GACxB,0BAA0B;;4FA8RjB,iCAAiC;kBA7S7C,SAAS;+BACE,8BAA8B,cAC5B,IAAI,WACP;wBACP,WAAW;wBACX,mBAAmB;wBACnB,kBAAkB;wBAClB,kBAAkB;wBAClB,cAAc;wBACd,eAAe;wBACf,iBAAiB;wBACjB,gBAAgB;wBAChB,mBAAmB;wBACnB,2BAA2B;wBAC3B,wBAAwB;wBACxB,0BAA0B;qBAC3B,mBACgB,uBAAuB,CAAC,MAAM,YACrC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwMT;8BA2FQ,KAAK;sBAAb,KAAK;gBAGG,eAAe;sBAAvB,KAAK;gBAGI,WAAW;sBAApB,MAAM","sourcesContent":["import {\r\n  Component,\r\n  ChangeDetectionStrategy,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  OnChanges,\r\n  OnDestroy,\r\n  SimpleChanges,\r\n  inject,\r\n} from '@angular/core';\r\nimport { FormBuilder, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subject } from 'rxjs';\r\nimport { debounceTime, takeUntil } from 'rxjs/operators';\r\n\r\nimport { UiFormFieldDescriptor, UiFieldType } from '../../../form-builder/types/index';\r\nimport { UiEditorAvailableField } from '../../types/editor.types';\r\nimport { UiValidationEditorComponent } from '../validation-editor/validation-editor.component';\r\nimport { UiOptionsEditorComponent } from '../options-editor/options-editor.component';\r\nimport { UiConditionEditorComponent } from '../condition-editor/condition-editor.component';\r\n\r\n/**\r\n * Tutti i tipi di campo supportati dal form builder.\r\n * Utilizzati per popolare la select del tipo nella configurazione campo.\r\n */\r\nconst FIELD_TYPES: { value: UiFieldType; label: string }[] = [\r\n  { value: 'text', label: 'Testo' },\r\n  { value: 'email', label: 'Email' },\r\n  { value: 'password', label: 'Password' },\r\n  { value: 'number', label: 'Numero' },\r\n  { value: 'textarea', label: 'Area di testo' },\r\n  { value: 'select', label: 'Select' },\r\n  { value: 'multiselect', label: 'Multiselect' },\r\n  { value: 'freemultiselect', label: 'Multiselect libero' },\r\n  { value: 'checkbox', label: 'Checkbox' },\r\n  { value: 'switch', label: 'Switch' },\r\n  { value: 'radio', label: 'Radio' },\r\n  { value: 'date', label: 'Data' },\r\n  { value: 'datetime', label: 'Data e ora' },\r\n  { value: 'file', label: 'File' },\r\n  { value: 'custom', label: 'Custom' },\r\n  { value: 'divider', label: 'Separatore' },\r\n  { value: 'flag', label: 'Flag' },\r\n  { value: 'location', label: 'Location' },\r\n  { value: 'location-table', label: 'Location Table' },\r\n];\r\n\r\n/**\r\n * Tipi di campo che supportano la configurazione delle opzioni.\r\n * Per questi tipi viene mostrato il pannello opzioni nell'editor.\r\n */\r\nconst OPTIONS_FIELD_TYPES: UiFieldType[] = ['select', 'multiselect', 'freemultiselect', 'radio'];\r\n\r\n/**\r\n * Pannello laterale destro per la configurazione di un campo selezionato.\r\n *\r\n * Mostra un form reattivo suddiviso in pannelli espandibili per gestire\r\n * proprieta base, stato, layout, validazioni, opzioni e condizioni.\r\n * Emette aggiornamenti parziali con debounce di 300ms.\r\n *\r\n * @selector ui-editor-field-config-panel\r\n */\r\n@Component({\r\n  selector: 'ui-editor-field-config-panel',\r\n  standalone: true,\r\n  imports: [\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    MatExpansionModule,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatCheckboxModule,\r\n    MatTooltipModule,\r\n    LucideAngularModule,\r\n    UiValidationEditorComponent,\r\n    UiOptionsEditorComponent,\r\n    UiConditionEditorComponent,\r\n  ],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"field-config-panel\">\r\n      <!-- Intestazione del pannello -->\r\n      <div class=\"field-config-panel__header\">\r\n        <lucide-icon name=\"settings\" [size]=\"18\" aria-hidden=\"true\" />\r\n        <span class=\"field-config-panel__title\">Configurazione Campo</span>\r\n      </div>\r\n\r\n      <!-- Stato vuoto: nessun campo selezionato -->\r\n      @if (!field) {\r\n        <div class=\"field-config-panel__empty\">\r\n          <lucide-icon name=\"mouse-pointer-click\" [size]=\"32\" aria-hidden=\"true\" />\r\n          <p class=\"field-config-panel__empty-text\">\r\n            Seleziona un campo per configurarlo\r\n          </p>\r\n        </div>\r\n      }\r\n\r\n      <!-- Form di configurazione del campo -->\r\n      @if (field && configForm) {\r\n        <div class=\"field-config-panel__content\">\r\n          <mat-accordion multi>\r\n\r\n            <!-- Pannello 1: Proprieta base -->\r\n            <mat-expansion-panel [expanded]=\"true\">\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"file-text\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Proprieta base</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <div class=\"field-config-panel__fields\" [formGroup]=\"configForm\">\r\n                <!-- Chiave univoca -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Chiave (key)</mat-label>\r\n                  <input matInput formControlName=\"key\" placeholder=\"campo_univoco\" />\r\n                  <mat-hint>Identificativo univoco del campo</mat-hint>\r\n                </mat-form-field>\r\n\r\n                <!-- Tipo di campo -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Tipo</mat-label>\r\n                  <mat-select formControlName=\"type\">\r\n                    @for (ft of fieldTypes; track ft.value) {\r\n                      <mat-option [value]=\"ft.value\">{{ ft.label }}</mat-option>\r\n                    }\r\n                  </mat-select>\r\n                </mat-form-field>\r\n\r\n                <!-- Label visualizzata -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Label</mat-label>\r\n                  <input matInput formControlName=\"label\" placeholder=\"Etichetta del campo\" />\r\n                </mat-form-field>\r\n\r\n                <!-- Placeholder -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Placeholder</mat-label>\r\n                  <input matInput formControlName=\"placeholder\" placeholder=\"Testo segnaposto\" />\r\n                </mat-form-field>\r\n\r\n                <!-- Tooltip -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Tooltip</mat-label>\r\n                  <input matInput formControlName=\"tooltip\" placeholder=\"Testo di aiuto\" />\r\n                </mat-form-field>\r\n\r\n                <!-- Valore di default -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Valore di default</mat-label>\r\n                  <input matInput formControlName=\"defaultValue\" placeholder=\"Valore iniziale\" />\r\n                </mat-form-field>\r\n              </div>\r\n            </mat-expansion-panel>\r\n\r\n            <!-- Pannello 2: Stato del campo -->\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"toggle-left\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Stato campo</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <div class=\"field-config-panel__fields\" [formGroup]=\"configForm\">\r\n                <mat-checkbox formControlName=\"required\" color=\"primary\">\r\n                  Obbligatorio\r\n                </mat-checkbox>\r\n\r\n                <mat-checkbox formControlName=\"disabled\" color=\"primary\">\r\n                  Disabilitato\r\n                </mat-checkbox>\r\n\r\n                <mat-checkbox formControlName=\"readonly\" color=\"primary\">\r\n                  Sola lettura\r\n                </mat-checkbox>\r\n\r\n                @if (isOptionsField) {\r\n                  <mat-checkbox formControlName=\"hideEmptyOption\" color=\"primary\">\r\n                    Nascondi opzione vuota\r\n                  </mat-checkbox>\r\n                }\r\n              </div>\r\n            </mat-expansion-panel>\r\n\r\n            <!-- Pannello 3: Layout -->\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"layout-grid\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Layout</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <div class=\"field-config-panel__fields\" [formGroup]=\"configForm\">\r\n                <!-- Colonne -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Colonne (1-12)</mat-label>\r\n                  <input matInput type=\"number\" formControlName=\"columns\" min=\"1\" max=\"12\" />\r\n                  <mat-hint>Numero di colonne nel grid</mat-hint>\r\n                </mat-form-field>\r\n\r\n                <!-- Ordine -->\r\n                <mat-form-field appearance=\"outline\" class=\"field-config-panel__field\">\r\n                  <mat-label>Ordine</mat-label>\r\n                  <input matInput type=\"number\" formControlName=\"order\" />\r\n                  <mat-hint>Posizione nel grid</mat-hint>\r\n                </mat-form-field>\r\n              </div>\r\n            </mat-expansion-panel>\r\n\r\n            <!-- Pannello 4: Validazioni -->\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"shield-check\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Validazioni</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <ui-validation-editor\r\n                [validations]=\"field.validation || []\"\r\n                (validationsChange)=\"onValidationsChange($event)\"\r\n              />\r\n            </mat-expansion-panel>\r\n\r\n            <!-- Pannello 5: Opzioni (solo per campi con opzioni) -->\r\n            @if (isOptionsField) {\r\n              <mat-expansion-panel>\r\n                <mat-expansion-panel-header>\r\n                  <mat-panel-title>\r\n                    <lucide-icon name=\"list\" [size]=\"16\" aria-hidden=\"true\" />\r\n                    <span>Opzioni</span>\r\n                  </mat-panel-title>\r\n                </mat-expansion-panel-header>\r\n\r\n                <ui-options-editor\r\n                  [options]=\"getFieldOptions()\"\r\n                  (optionsChange)=\"onOptionsChange($event)\"\r\n                />\r\n              </mat-expansion-panel>\r\n            }\r\n\r\n            <!-- Pannello 6: Condizioni di visibilita -->\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"eye\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Condizioni di visibilita</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <ui-condition-editor\r\n                [conditions]=\"field.conditions || []\"\r\n                [availableFields]=\"availableFields\"\r\n                (conditionsChange)=\"onVisibilityConditionsChange($event)\"\r\n              />\r\n            </mat-expansion-panel>\r\n\r\n            <!-- Pannello 7: Condizioni di disabilitazione -->\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>\r\n                  <lucide-icon name=\"ban\" [size]=\"16\" aria-hidden=\"true\" />\r\n                  <span>Condizioni di disabilitazione</span>\r\n                </mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n\r\n              <ui-condition-editor\r\n                [conditions]=\"field.disableConditions || []\"\r\n                [availableFields]=\"availableFields\"\r\n                (conditionsChange)=\"onDisableConditionsChange($event)\"\r\n              />\r\n            </mat-expansion-panel>\r\n\r\n          </mat-accordion>\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n    /* Contenitore principale del pannello di configurazione */\r\n    .field-config-panel {\r\n      display: flex;\r\n      flex-direction: column;\r\n      height: 100%;\r\n      background: var(--ui-color-surface, #fff);\r\n      border-left: 1px solid var(--ui-color-border, #e0e0e0);\r\n      overflow-y: auto;\r\n    }\r\n\r\n    /* Intestazione del pannello */\r\n    .field-config-panel__header {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n      padding: var(--ui-spacing-3, 12px) var(--ui-spacing-4, 16px);\r\n      border-bottom: 1px solid var(--ui-color-border, #e0e0e0);\r\n      background: var(--ui-color-bg-subtle, #fafafa);\r\n    }\r\n\r\n    .field-config-panel__title {\r\n      font-weight: 600;\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      color: var(--ui-color-text, #212121);\r\n    }\r\n\r\n    /* Stato vuoto: nessun campo selezionato */\r\n    .field-config-panel__empty {\r\n      display: flex;\r\n      flex-direction: column;\r\n      align-items: center;\r\n      justify-content: center;\r\n      gap: var(--ui-spacing-3, 12px);\r\n      padding: var(--ui-spacing-8, 64px) var(--ui-spacing-4, 16px);\r\n      text-align: center;\r\n      color: var(--ui-color-text-muted, #9e9e9e);\r\n    }\r\n\r\n    .field-config-panel__empty-text {\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      margin: 0;\r\n    }\r\n\r\n    /* Contenuto scrollabile */\r\n    .field-config-panel__content {\r\n      flex: 1;\r\n      overflow-y: auto;\r\n      padding: var(--ui-spacing-2, 8px);\r\n    }\r\n\r\n    /* Gruppo di campi dentro un pannello */\r\n    .field-config-panel__fields {\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: var(--ui-spacing-2, 8px);\r\n      padding: var(--ui-spacing-1, 4px) 0;\r\n    }\r\n\r\n    /* Singolo campo del form di configurazione */\r\n    .field-config-panel__field {\r\n      width: 100%;\r\n    }\r\n\r\n    /* Stili per i titoli dei pannelli espandibili */\r\n    mat-panel-title {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: var(--ui-spacing-2, 8px);\r\n      font-size: var(--ui-font-size-sm, 0.875rem);\r\n      font-weight: 500;\r\n      color: var(--ui-color-text, #212121);\r\n    }\r\n\r\n    /* Checkbox con margine superiore */\r\n    mat-checkbox {\r\n      margin: var(--ui-spacing-1, 4px) 0;\r\n    }\r\n  `,\r\n  ],\r\n})\r\nexport class UiEditorFieldConfigPanelComponent implements OnChanges, OnDestroy {\r\n  /** @internal Servizio per la costruzione del FormGroup. */\r\n  private readonly fb = inject(FormBuilder);\r\n\r\n  /** @internal Subject per la distruzione e pulizia delle sottoscrizioni. */\r\n  private readonly destroy$ = new Subject<void>();\r\n\r\n  /** Campo attualmente selezionato per la configurazione. */\r\n  @Input() field: UiFormFieldDescriptor | null = null;\r\n\r\n  /** Lista dei campi disponibili per le condizioni. */\r\n  @Input() availableFields: UiEditorAvailableField[] = [];\r\n\r\n  /** Emesso quando il campo viene aggiornato dall'utente. */\r\n  @Output() fieldUpdate = new EventEmitter<Partial<UiFormFieldDescriptor>>();\r\n\r\n  /** Form reattivo per la configurazione del campo. */\r\n  configForm: FormGroup | null = null;\r\n\r\n  /** Lista dei tipi di campo per la select. */\r\n  readonly fieldTypes = FIELD_TYPES;\r\n\r\n  /**\r\n   * Indica se il campo selezionato supporta le opzioni.\r\n   * Viene utilizzato per mostrare/nascondere il pannello opzioni.\r\n   */\r\n  get isOptionsField(): boolean {\r\n    if (!this.field) return false;\r\n    return OPTIONS_FIELD_TYPES.includes(this.field.type);\r\n  }\r\n\r\n  /**\r\n   * Rileva cambiamenti sugli input e ricostruisce il form\r\n   * quando cambia il campo selezionato.\r\n   */\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['field']) {\r\n      this.buildForm();\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  /**\r\n   * Restituisce le opzioni del campo come array statico.\r\n   * Ignora opzioni osservabili poiche non gestibili nell'editor.\r\n   */\r\n  getFieldOptions(): { value: any; label: string }[] {\r\n    if (!this.field?.options || !Array.isArray(this.field.options)) return [];\r\n    return this.field.options.map((o) => ({ value: o.value, label: o.label }));\r\n  }\r\n\r\n  /**\r\n   * Gestisce l'aggiornamento delle validazioni dal sotto-componente.\r\n   * @param validations - Nuova lista di regole di validazione\r\n   */\r\n  onValidationsChange(validations: any[]): void {\r\n    this.fieldUpdate.emit({ validation: validations });\r\n  }\r\n\r\n  /**\r\n   * Gestisce l'aggiornamento delle opzioni dal sotto-componente.\r\n   * @param options - Nuova lista di opzioni\r\n   */\r\n  onOptionsChange(options: { value: any; label: string }[]): void {\r\n    this.fieldUpdate.emit({ options });\r\n  }\r\n\r\n  /**\r\n   * Gestisce l'aggiornamento delle condizioni di visibilita.\r\n   * @param conditions - Nuova lista di condizioni\r\n   */\r\n  onVisibilityConditionsChange(conditions: any[]): void {\r\n    this.fieldUpdate.emit({ conditions });\r\n  }\r\n\r\n  /**\r\n   * Gestisce l'aggiornamento delle condizioni di disabilitazione.\r\n   * @param conditions - Nuova lista di condizioni\r\n   */\r\n  onDisableConditionsChange(conditions: any[]): void {\r\n    this.fieldUpdate.emit({ disableConditions: conditions });\r\n  }\r\n\r\n  /**\r\n   * @internal\r\n   * Costruisce il FormGroup a partire dal campo selezionato.\r\n   * Si sottoscrive ai valueChanges con debounce per emettere aggiornamenti.\r\n   */\r\n  private buildForm(): void {\r\n    this.destroy$.next();\r\n\r\n    if (!this.field) {\r\n      this.configForm = null;\r\n      return;\r\n    }\r\n\r\n    this.configForm = this.fb.group({\r\n      key: [this.field.key],\r\n      type: [this.field.type],\r\n      label: [this.field.label],\r\n      placeholder: [this.field.placeholder || ''],\r\n      tooltip: [this.field.tooltip || ''],\r\n      defaultValue: [this.field.defaultValue ?? ''],\r\n      required: [this.field.required || false],\r\n      disabled: [this.field.disabled || false],\r\n      readonly: [this.field.readonly || false],\r\n      hideEmptyOption: [this.field.hideEmptyOption || false],\r\n      columns: [this.field.layout?.columns ?? null],\r\n      order: [this.field.layout?.order ?? null],\r\n    });\r\n\r\n    this.configForm.valueChanges.pipe(debounceTime(300), takeUntil(this.destroy$)).subscribe((values) => {\r\n      /** Mappa i valori del form nel formato di aggiornamento parziale del campo */\r\n      const update: Partial<UiFormFieldDescriptor> = {\r\n        key: values.key,\r\n        type: values.type,\r\n        label: values.label,\r\n        placeholder: values.placeholder || undefined,\r\n        tooltip: values.tooltip || undefined,\r\n        defaultValue: values.defaultValue || undefined,\r\n        required: values.required,\r\n        disabled: values.disabled,\r\n        readonly: values.readonly,\r\n        hideEmptyOption: values.hideEmptyOption,\r\n        layout: {\r\n          columns: values.columns ?? undefined,\r\n          order: values.order ?? undefined,\r\n        },\r\n      };\r\n      this.fieldUpdate.emit(update);\r\n    });\r\n  }\r\n}\r\n"]}
|