@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,324 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/form-builder-editor
|
|
3
|
+
* Servizio per la gestione centralizzata dello stato dell'editor.
|
|
4
|
+
*/
|
|
5
|
+
import { Injectable, inject } from '@angular/core';
|
|
6
|
+
import { BehaviorSubject } from 'rxjs';
|
|
7
|
+
import { UiEditorFieldFactoryService } from './field-factory.service';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
/**
|
|
10
|
+
* Servizio per la gestione reattiva dello stato dell'editor visuale.
|
|
11
|
+
*
|
|
12
|
+
* Espone lo stato come Observable e fornisce metodi immutabili
|
|
13
|
+
* per tutte le operazioni CRUD su sezioni e campi.
|
|
14
|
+
*/
|
|
15
|
+
export class UiEditorStateService {
|
|
16
|
+
constructor() {
|
|
17
|
+
/** @internal Factory per la creazione dei campi. */
|
|
18
|
+
this.fieldFactory = inject(UiEditorFieldFactoryService);
|
|
19
|
+
/** @internal Stato iniziale dell'editor. */
|
|
20
|
+
this.initialState = {
|
|
21
|
+
schema: {
|
|
22
|
+
id: `form_${Date.now()}`,
|
|
23
|
+
title: 'Nuovo Form',
|
|
24
|
+
description: '',
|
|
25
|
+
sections: [],
|
|
26
|
+
},
|
|
27
|
+
selectedSectionId: null,
|
|
28
|
+
selectedFieldKey: null,
|
|
29
|
+
isDirty: false,
|
|
30
|
+
lastSaved: null,
|
|
31
|
+
};
|
|
32
|
+
/** @internal Subject per lo stato dell'editor. */
|
|
33
|
+
this._state$ = new BehaviorSubject(this.initialState);
|
|
34
|
+
/** Stream reattivo dello stato dell'editor. */
|
|
35
|
+
this.state$ = this._state$.asObservable();
|
|
36
|
+
}
|
|
37
|
+
// ─── Lettura stato ───────────────────────────────────────────
|
|
38
|
+
/** Restituisce un snapshot dello stato corrente. */
|
|
39
|
+
getCurrentState() {
|
|
40
|
+
return this._state$.value;
|
|
41
|
+
}
|
|
42
|
+
/** Restituisce lo schema corrente. */
|
|
43
|
+
getSchema() {
|
|
44
|
+
return this.getCurrentState().schema;
|
|
45
|
+
}
|
|
46
|
+
// ─── Aggiornamento stato ─────────────────────────────────────
|
|
47
|
+
/** @internal Aggiorna lo stato e marca come dirty. */
|
|
48
|
+
updateState(updates) {
|
|
49
|
+
this._state$.next({
|
|
50
|
+
...this.getCurrentState(),
|
|
51
|
+
...updates,
|
|
52
|
+
isDirty: true,
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
/** Imposta uno schema completo (es. da import). */
|
|
56
|
+
setSchema(schema) {
|
|
57
|
+
this.updateState({ schema });
|
|
58
|
+
}
|
|
59
|
+
/** Marca lo stato come salvato. */
|
|
60
|
+
markAsSaved() {
|
|
61
|
+
this._state$.next({
|
|
62
|
+
...this.getCurrentState(),
|
|
63
|
+
isDirty: false,
|
|
64
|
+
lastSaved: new Date(),
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
/** Reset allo stato iniziale. */
|
|
68
|
+
reset() {
|
|
69
|
+
this._state$.next({
|
|
70
|
+
...this.initialState,
|
|
71
|
+
schema: {
|
|
72
|
+
id: `form_${Date.now()}`,
|
|
73
|
+
title: 'Nuovo Form',
|
|
74
|
+
description: '',
|
|
75
|
+
sections: [],
|
|
76
|
+
},
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
/** Pulisce lo schema mantenendo l'ID. */
|
|
80
|
+
clearSchema() {
|
|
81
|
+
const currentId = this.getSchema().id;
|
|
82
|
+
this.updateState({
|
|
83
|
+
schema: { id: currentId, sections: [] },
|
|
84
|
+
selectedSectionId: null,
|
|
85
|
+
selectedFieldKey: null,
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
// ─── Operazioni su sezioni ───────────────────────────────────
|
|
89
|
+
/** Aggiunge una nuova sezione e la seleziona. */
|
|
90
|
+
addSection(section) {
|
|
91
|
+
const schema = { ...this.getSchema() };
|
|
92
|
+
const sectionId = section?.id || `section_${Date.now()}`;
|
|
93
|
+
const newSection = {
|
|
94
|
+
id: sectionId,
|
|
95
|
+
title: section?.title || `Sezione ${schema.sections.length + 1}`,
|
|
96
|
+
description: section?.description || '',
|
|
97
|
+
fields: section?.fields || [],
|
|
98
|
+
collapsible: false,
|
|
99
|
+
collapsed: false,
|
|
100
|
+
};
|
|
101
|
+
schema.sections = [...schema.sections, newSection];
|
|
102
|
+
this.updateState({ schema, selectedSectionId: sectionId });
|
|
103
|
+
return sectionId;
|
|
104
|
+
}
|
|
105
|
+
/** Rimuove una sezione per ID. */
|
|
106
|
+
removeSection(sectionId) {
|
|
107
|
+
const schema = { ...this.getSchema() };
|
|
108
|
+
schema.sections = schema.sections.filter((s) => s.id !== sectionId);
|
|
109
|
+
const current = this.getCurrentState();
|
|
110
|
+
const updates = { schema };
|
|
111
|
+
if (current.selectedSectionId === sectionId) {
|
|
112
|
+
updates.selectedSectionId = null;
|
|
113
|
+
updates.selectedFieldKey = null;
|
|
114
|
+
}
|
|
115
|
+
this.updateState(updates);
|
|
116
|
+
}
|
|
117
|
+
/** Aggiorna proprieta di una sezione. */
|
|
118
|
+
updateSection(sectionId, updates) {
|
|
119
|
+
const schema = { ...this.getSchema() };
|
|
120
|
+
schema.sections = schema.sections.map((s) => (s.id === sectionId ? { ...s, ...updates } : s));
|
|
121
|
+
this.updateState({ schema });
|
|
122
|
+
}
|
|
123
|
+
/** Duplica una sezione con nuovi ID per sezione e campi. */
|
|
124
|
+
duplicateSection(sectionId) {
|
|
125
|
+
const schema = { ...this.getSchema() };
|
|
126
|
+
const section = schema.sections.find((s) => s.id === sectionId);
|
|
127
|
+
if (!section)
|
|
128
|
+
return null;
|
|
129
|
+
const newSectionId = `section_${Date.now()}`;
|
|
130
|
+
const duplicated = {
|
|
131
|
+
...JSON.parse(JSON.stringify(section)),
|
|
132
|
+
id: newSectionId,
|
|
133
|
+
title: `${section.title} (Copia)`,
|
|
134
|
+
fields: section.fields.map((f) => this.fieldFactory.duplicateField(f)),
|
|
135
|
+
};
|
|
136
|
+
const index = schema.sections.findIndex((s) => s.id === sectionId);
|
|
137
|
+
schema.sections = [...schema.sections.slice(0, index + 1), duplicated, ...schema.sections.slice(index + 1)];
|
|
138
|
+
this.updateState({ schema, selectedSectionId: newSectionId });
|
|
139
|
+
return newSectionId;
|
|
140
|
+
}
|
|
141
|
+
// ─── Operazioni su campi ─────────────────────────────────────
|
|
142
|
+
/** Aggiunge un campo a una sezione in una posizione specifica. */
|
|
143
|
+
addFieldAtIndex(sectionId, field, index) {
|
|
144
|
+
const schema = { ...this.getSchema() };
|
|
145
|
+
schema.sections = schema.sections.map((s) => {
|
|
146
|
+
if (s.id !== sectionId)
|
|
147
|
+
return s;
|
|
148
|
+
const fields = [...s.fields];
|
|
149
|
+
fields.splice(index, 0, field);
|
|
150
|
+
return { ...s, fields };
|
|
151
|
+
});
|
|
152
|
+
this.updateState({ schema, selectedFieldKey: field.key });
|
|
153
|
+
}
|
|
154
|
+
/** Rimuove un campo da una sezione. */
|
|
155
|
+
removeField(sectionId, fieldKey) {
|
|
156
|
+
const schema = { ...this.getSchema() };
|
|
157
|
+
schema.sections = schema.sections.map((s) => {
|
|
158
|
+
if (s.id !== sectionId)
|
|
159
|
+
return s;
|
|
160
|
+
return { ...s, fields: s.fields.filter((f) => f.key !== fieldKey) };
|
|
161
|
+
});
|
|
162
|
+
const current = this.getCurrentState();
|
|
163
|
+
const updates = { schema };
|
|
164
|
+
if (current.selectedFieldKey === fieldKey) {
|
|
165
|
+
updates.selectedFieldKey = null;
|
|
166
|
+
}
|
|
167
|
+
this.updateState(updates);
|
|
168
|
+
}
|
|
169
|
+
/** Aggiorna proprieta di un campo. */
|
|
170
|
+
updateField(sectionId, fieldKey, fieldUpdates) {
|
|
171
|
+
const schema = { ...this.getSchema() };
|
|
172
|
+
schema.sections = schema.sections.map((s) => {
|
|
173
|
+
if (s.id !== sectionId)
|
|
174
|
+
return s;
|
|
175
|
+
return {
|
|
176
|
+
...s,
|
|
177
|
+
fields: s.fields.map((f) => (f.key === fieldKey ? { ...f, ...fieldUpdates } : f)),
|
|
178
|
+
};
|
|
179
|
+
});
|
|
180
|
+
this.updateState({ schema });
|
|
181
|
+
}
|
|
182
|
+
/** Duplica un campo nella stessa sezione. */
|
|
183
|
+
duplicateField(sectionId, fieldKey) {
|
|
184
|
+
const schema = { ...this.getSchema() };
|
|
185
|
+
schema.sections = schema.sections.map((s) => {
|
|
186
|
+
if (s.id !== sectionId)
|
|
187
|
+
return s;
|
|
188
|
+
const field = s.fields.find((f) => f.key === fieldKey);
|
|
189
|
+
if (!field)
|
|
190
|
+
return s;
|
|
191
|
+
const duplicated = this.fieldFactory.duplicateField(field);
|
|
192
|
+
const index = s.fields.findIndex((f) => f.key === fieldKey);
|
|
193
|
+
const fields = [...s.fields];
|
|
194
|
+
fields.splice(index + 1, 0, duplicated);
|
|
195
|
+
return { ...s, fields };
|
|
196
|
+
});
|
|
197
|
+
this.updateState({ schema });
|
|
198
|
+
}
|
|
199
|
+
/** Riordina un campo all'interno della stessa sezione. */
|
|
200
|
+
moveFieldInSection(sectionId, fromIndex, toIndex) {
|
|
201
|
+
const schema = { ...this.getSchema() };
|
|
202
|
+
schema.sections = schema.sections.map((s) => {
|
|
203
|
+
if (s.id !== sectionId)
|
|
204
|
+
return s;
|
|
205
|
+
const fields = [...s.fields];
|
|
206
|
+
const [moved] = fields.splice(fromIndex, 1);
|
|
207
|
+
fields.splice(toIndex, 0, moved);
|
|
208
|
+
return { ...s, fields };
|
|
209
|
+
});
|
|
210
|
+
this.updateState({ schema });
|
|
211
|
+
}
|
|
212
|
+
/** Sposta un campo tra due sezioni diverse. */
|
|
213
|
+
moveFieldBetweenSections(fromSectionId, toSectionId, fieldKey, toIndex) {
|
|
214
|
+
const schema = { ...this.getSchema() };
|
|
215
|
+
let movedField;
|
|
216
|
+
schema.sections = schema.sections.map((s) => {
|
|
217
|
+
if (s.id === fromSectionId) {
|
|
218
|
+
const idx = s.fields.findIndex((f) => f.key === fieldKey);
|
|
219
|
+
if (idx !== -1) {
|
|
220
|
+
const fields = [...s.fields];
|
|
221
|
+
[movedField] = fields.splice(idx, 1);
|
|
222
|
+
return { ...s, fields };
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
return s;
|
|
226
|
+
});
|
|
227
|
+
if (movedField) {
|
|
228
|
+
schema.sections = schema.sections.map((s) => {
|
|
229
|
+
if (s.id === toSectionId) {
|
|
230
|
+
const fields = [...s.fields];
|
|
231
|
+
fields.splice(toIndex, 0, movedField);
|
|
232
|
+
return { ...s, fields };
|
|
233
|
+
}
|
|
234
|
+
return s;
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
this.updateState({ schema });
|
|
238
|
+
}
|
|
239
|
+
// ─── Selezione ───────────────────────────────────────────────
|
|
240
|
+
/** Seleziona una sezione (deseleziona il campo). */
|
|
241
|
+
selectSection(sectionId) {
|
|
242
|
+
this._state$.next({
|
|
243
|
+
...this.getCurrentState(),
|
|
244
|
+
selectedSectionId: sectionId,
|
|
245
|
+
selectedFieldKey: null,
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
/** Seleziona un campo (e la sua sezione). */
|
|
249
|
+
selectField(sectionId, fieldKey) {
|
|
250
|
+
this._state$.next({
|
|
251
|
+
...this.getCurrentState(),
|
|
252
|
+
selectedSectionId: sectionId,
|
|
253
|
+
selectedFieldKey: fieldKey,
|
|
254
|
+
});
|
|
255
|
+
}
|
|
256
|
+
// ─── Ricerca ─────────────────────────────────────────────────
|
|
257
|
+
/** Trova la sezione che contiene un campo. */
|
|
258
|
+
findSectionByFieldKey(fieldKey) {
|
|
259
|
+
return this.getSchema().sections.find((s) => s.fields.some((f) => f.key === fieldKey)) || null;
|
|
260
|
+
}
|
|
261
|
+
/** Trova un campo per chiave attraverso tutte le sezioni. */
|
|
262
|
+
findFieldByKey(fieldKey) {
|
|
263
|
+
const section = this.findSectionByFieldKey(fieldKey);
|
|
264
|
+
return section?.fields.find((f) => f.key === fieldKey) || null;
|
|
265
|
+
}
|
|
266
|
+
// ─── Validazione ─────────────────────────────────────────────
|
|
267
|
+
/** Valida lo schema corrente cercando errori e warning strutturali. */
|
|
268
|
+
validateSchema() {
|
|
269
|
+
const schema = this.getSchema();
|
|
270
|
+
const errors = [];
|
|
271
|
+
const warnings = [];
|
|
272
|
+
const allKeys = new Set();
|
|
273
|
+
schema.sections.forEach((section, si) => {
|
|
274
|
+
section.fields.forEach((field, fi) => {
|
|
275
|
+
// Chiavi duplicate
|
|
276
|
+
if (allKeys.has(field.key)) {
|
|
277
|
+
errors.push({
|
|
278
|
+
type: 'duplicate_key',
|
|
279
|
+
message: `Chiave duplicata: ${field.key}`,
|
|
280
|
+
path: `sections[${si}].fields[${fi}].key`,
|
|
281
|
+
});
|
|
282
|
+
}
|
|
283
|
+
allKeys.add(field.key);
|
|
284
|
+
// Label mancante
|
|
285
|
+
if (!field.label?.trim()) {
|
|
286
|
+
warnings.push({
|
|
287
|
+
type: 'missing_label',
|
|
288
|
+
message: `Campo senza label: ${field.key}`,
|
|
289
|
+
path: `sections[${si}].fields[${fi}].label`,
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
// Opzioni vuote per campi di scelta
|
|
293
|
+
if (['select', 'radio', 'multiselect'].includes(field.type)) {
|
|
294
|
+
if (!field.options || (Array.isArray(field.options) && field.options.length === 0)) {
|
|
295
|
+
warnings.push({
|
|
296
|
+
type: 'empty_options',
|
|
297
|
+
message: `Campo ${field.key} di tipo ${field.type} senza opzioni`,
|
|
298
|
+
path: `sections[${si}].fields[${fi}].options`,
|
|
299
|
+
});
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
// Riferimenti nelle condizioni
|
|
303
|
+
if (field.conditions) {
|
|
304
|
+
field.conditions.forEach((cond, ci) => {
|
|
305
|
+
if (cond.field && !allKeys.has(cond.field)) {
|
|
306
|
+
errors.push({
|
|
307
|
+
type: 'invalid_reference',
|
|
308
|
+
message: `Riferimento a campo inesistente: ${cond.field} in ${field.key}`,
|
|
309
|
+
path: `sections[${si}].fields[${fi}].conditions[${ci}].field`,
|
|
310
|
+
});
|
|
311
|
+
}
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
});
|
|
315
|
+
});
|
|
316
|
+
return { valid: errors.length === 0, errors, warnings };
|
|
317
|
+
}
|
|
318
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
319
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorStateService }); }
|
|
320
|
+
}
|
|
321
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorStateService, decorators: [{
|
|
322
|
+
type: Injectable
|
|
323
|
+
}] });
|
|
324
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor-state.service.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/services/editor-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/form-builder-editor\r\n * Servizio per la gestione centralizzata dello stato dell'editor.\r\n */\r\n\r\nimport { Injectable, inject } from '@angular/core';\r\nimport { BehaviorSubject, Observable } from 'rxjs';\r\nimport { UiFormSchema, UiFormSection, UiFormFieldDescriptor } from '../../form-builder/types/index';\r\nimport {\r\n  UiEditorState,\r\n  UiEditorSchemaValidationResult,\r\n  UiEditorSchemaValidationError,\r\n  UiEditorSchemaValidationWarning,\r\n} from '../types/editor.types';\r\nimport { UiEditorFieldFactoryService } from './field-factory.service';\r\n\r\n/**\r\n * Servizio per la gestione reattiva dello stato dell'editor visuale.\r\n *\r\n * Espone lo stato come Observable e fornisce metodi immutabili\r\n * per tutte le operazioni CRUD su sezioni e campi.\r\n */\r\n@Injectable()\r\nexport class UiEditorStateService {\r\n  /** @internal Factory per la creazione dei campi. */\r\n  private readonly fieldFactory = inject(UiEditorFieldFactoryService);\r\n\r\n  /** @internal Stato iniziale dell'editor. */\r\n  private readonly initialState: UiEditorState = {\r\n    schema: {\r\n      id: `form_${Date.now()}`,\r\n      title: 'Nuovo Form',\r\n      description: '',\r\n      sections: [],\r\n    },\r\n    selectedSectionId: null,\r\n    selectedFieldKey: null,\r\n    isDirty: false,\r\n    lastSaved: null,\r\n  };\r\n\r\n  /** @internal Subject per lo stato dell'editor. */\r\n  private readonly _state$ = new BehaviorSubject<UiEditorState>(this.initialState);\r\n\r\n  /** Stream reattivo dello stato dell'editor. */\r\n  readonly state$: Observable<UiEditorState> = this._state$.asObservable();\r\n\r\n  // ─── Lettura stato ───────────────────────────────────────────\r\n\r\n  /** Restituisce un snapshot dello stato corrente. */\r\n  getCurrentState(): UiEditorState {\r\n    return this._state$.value;\r\n  }\r\n\r\n  /** Restituisce lo schema corrente. */\r\n  getSchema(): UiFormSchema {\r\n    return this.getCurrentState().schema;\r\n  }\r\n\r\n  // ─── Aggiornamento stato ─────────────────────────────────────\r\n\r\n  /** @internal Aggiorna lo stato e marca come dirty. */\r\n  private updateState(updates: Partial<UiEditorState>): void {\r\n    this._state$.next({\r\n      ...this.getCurrentState(),\r\n      ...updates,\r\n      isDirty: true,\r\n    });\r\n  }\r\n\r\n  /** Imposta uno schema completo (es. da import). */\r\n  setSchema(schema: UiFormSchema): void {\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  /** Marca lo stato come salvato. */\r\n  markAsSaved(): void {\r\n    this._state$.next({\r\n      ...this.getCurrentState(),\r\n      isDirty: false,\r\n      lastSaved: new Date(),\r\n    });\r\n  }\r\n\r\n  /** Reset allo stato iniziale. */\r\n  reset(): void {\r\n    this._state$.next({\r\n      ...this.initialState,\r\n      schema: {\r\n        id: `form_${Date.now()}`,\r\n        title: 'Nuovo Form',\r\n        description: '',\r\n        sections: [],\r\n      },\r\n    });\r\n  }\r\n\r\n  /** Pulisce lo schema mantenendo l'ID. */\r\n  clearSchema(): void {\r\n    const currentId = this.getSchema().id;\r\n    this.updateState({\r\n      schema: { id: currentId, sections: [] },\r\n      selectedSectionId: null,\r\n      selectedFieldKey: null,\r\n    });\r\n  }\r\n\r\n  // ─── Operazioni su sezioni ───────────────────────────────────\r\n\r\n  /** Aggiunge una nuova sezione e la seleziona. */\r\n  addSection(section?: Partial<UiFormSection>): string {\r\n    const schema = { ...this.getSchema() };\r\n    const sectionId = section?.id || `section_${Date.now()}`;\r\n\r\n    const newSection: UiFormSection = {\r\n      id: sectionId,\r\n      title: section?.title || `Sezione ${schema.sections.length + 1}`,\r\n      description: section?.description || '',\r\n      fields: section?.fields || [],\r\n      collapsible: false,\r\n      collapsed: false,\r\n    };\r\n\r\n    schema.sections = [...schema.sections, newSection];\r\n    this.updateState({ schema, selectedSectionId: sectionId });\r\n    return sectionId;\r\n  }\r\n\r\n  /** Rimuove una sezione per ID. */\r\n  removeSection(sectionId: string): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.filter((s) => s.id !== sectionId);\r\n\r\n    const current = this.getCurrentState();\r\n    const updates: Partial<UiEditorState> = { schema };\r\n\r\n    if (current.selectedSectionId === sectionId) {\r\n      updates.selectedSectionId = null;\r\n      updates.selectedFieldKey = null;\r\n    }\r\n\r\n    this.updateState(updates);\r\n  }\r\n\r\n  /** Aggiorna proprieta di una sezione. */\r\n  updateSection(sectionId: string, updates: Partial<UiFormSection>): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => (s.id === sectionId ? { ...s, ...updates } : s));\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  /** Duplica una sezione con nuovi ID per sezione e campi. */\r\n  duplicateSection(sectionId: string): string | null {\r\n    const schema = { ...this.getSchema() };\r\n    const section = schema.sections.find((s) => s.id === sectionId);\r\n    if (!section) return null;\r\n\r\n    const newSectionId = `section_${Date.now()}`;\r\n    const duplicated: UiFormSection = {\r\n      ...JSON.parse(JSON.stringify(section)),\r\n      id: newSectionId,\r\n      title: `${section.title} (Copia)`,\r\n      fields: section.fields.map((f) => this.fieldFactory.duplicateField(f)),\r\n    };\r\n\r\n    const index = schema.sections.findIndex((s) => s.id === sectionId);\r\n    schema.sections = [...schema.sections.slice(0, index + 1), duplicated, ...schema.sections.slice(index + 1)];\r\n\r\n    this.updateState({ schema, selectedSectionId: newSectionId });\r\n    return newSectionId;\r\n  }\r\n\r\n  // ─── Operazioni su campi ─────────────────────────────────────\r\n\r\n  /** Aggiunge un campo a una sezione in una posizione specifica. */\r\n  addFieldAtIndex(sectionId: string, field: UiFormFieldDescriptor, index: number): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id !== sectionId) return s;\r\n      const fields = [...s.fields];\r\n      fields.splice(index, 0, field);\r\n      return { ...s, fields };\r\n    });\r\n    this.updateState({ schema, selectedFieldKey: field.key });\r\n  }\r\n\r\n  /** Rimuove un campo da una sezione. */\r\n  removeField(sectionId: string, fieldKey: string): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id !== sectionId) return s;\r\n      return { ...s, fields: s.fields.filter((f) => f.key !== fieldKey) };\r\n    });\r\n\r\n    const current = this.getCurrentState();\r\n    const updates: Partial<UiEditorState> = { schema };\r\n    if (current.selectedFieldKey === fieldKey) {\r\n      updates.selectedFieldKey = null;\r\n    }\r\n    this.updateState(updates);\r\n  }\r\n\r\n  /** Aggiorna proprieta di un campo. */\r\n  updateField(sectionId: string, fieldKey: string, fieldUpdates: Partial<UiFormFieldDescriptor>): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id !== sectionId) return s;\r\n      return {\r\n        ...s,\r\n        fields: s.fields.map((f) => (f.key === fieldKey ? { ...f, ...fieldUpdates } : f)),\r\n      };\r\n    });\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  /** Duplica un campo nella stessa sezione. */\r\n  duplicateField(sectionId: string, fieldKey: string): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id !== sectionId) return s;\r\n      const field = s.fields.find((f) => f.key === fieldKey);\r\n      if (!field) return s;\r\n\r\n      const duplicated = this.fieldFactory.duplicateField(field);\r\n      const index = s.fields.findIndex((f) => f.key === fieldKey);\r\n      const fields = [...s.fields];\r\n      fields.splice(index + 1, 0, duplicated);\r\n      return { ...s, fields };\r\n    });\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  /** Riordina un campo all'interno della stessa sezione. */\r\n  moveFieldInSection(sectionId: string, fromIndex: number, toIndex: number): void {\r\n    const schema = { ...this.getSchema() };\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id !== sectionId) return s;\r\n      const fields = [...s.fields];\r\n      const [moved] = fields.splice(fromIndex, 1);\r\n      fields.splice(toIndex, 0, moved);\r\n      return { ...s, fields };\r\n    });\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  /** Sposta un campo tra due sezioni diverse. */\r\n  moveFieldBetweenSections(fromSectionId: string, toSectionId: string, fieldKey: string, toIndex: number): void {\r\n    const schema = { ...this.getSchema() };\r\n    let movedField: UiFormFieldDescriptor | undefined;\r\n\r\n    schema.sections = schema.sections.map((s) => {\r\n      if (s.id === fromSectionId) {\r\n        const idx = s.fields.findIndex((f) => f.key === fieldKey);\r\n        if (idx !== -1) {\r\n          const fields = [...s.fields];\r\n          [movedField] = fields.splice(idx, 1);\r\n          return { ...s, fields };\r\n        }\r\n      }\r\n      return s;\r\n    });\r\n\r\n    if (movedField) {\r\n      schema.sections = schema.sections.map((s) => {\r\n        if (s.id === toSectionId) {\r\n          const fields = [...s.fields];\r\n          fields.splice(toIndex, 0, movedField!);\r\n          return { ...s, fields };\r\n        }\r\n        return s;\r\n      });\r\n    }\r\n\r\n    this.updateState({ schema });\r\n  }\r\n\r\n  // ─── Selezione ───────────────────────────────────────────────\r\n\r\n  /** Seleziona una sezione (deseleziona il campo). */\r\n  selectSection(sectionId: string | null): void {\r\n    this._state$.next({\r\n      ...this.getCurrentState(),\r\n      selectedSectionId: sectionId,\r\n      selectedFieldKey: null,\r\n    });\r\n  }\r\n\r\n  /** Seleziona un campo (e la sua sezione). */\r\n  selectField(sectionId: string, fieldKey: string | null): void {\r\n    this._state$.next({\r\n      ...this.getCurrentState(),\r\n      selectedSectionId: sectionId,\r\n      selectedFieldKey: fieldKey,\r\n    });\r\n  }\r\n\r\n  // ─── Ricerca ─────────────────────────────────────────────────\r\n\r\n  /** Trova la sezione che contiene un campo. */\r\n  findSectionByFieldKey(fieldKey: string): UiFormSection | null {\r\n    return this.getSchema().sections.find((s) => s.fields.some((f) => f.key === fieldKey)) || null;\r\n  }\r\n\r\n  /** Trova un campo per chiave attraverso tutte le sezioni. */\r\n  findFieldByKey(fieldKey: string): UiFormFieldDescriptor | null {\r\n    const section = this.findSectionByFieldKey(fieldKey);\r\n    return section?.fields.find((f) => f.key === fieldKey) || null;\r\n  }\r\n\r\n  // ─── Validazione ─────────────────────────────────────────────\r\n\r\n  /** Valida lo schema corrente cercando errori e warning strutturali. */\r\n  validateSchema(): UiEditorSchemaValidationResult {\r\n    const schema = this.getSchema();\r\n    const errors: UiEditorSchemaValidationError[] = [];\r\n    const warnings: UiEditorSchemaValidationWarning[] = [];\r\n    const allKeys = new Set<string>();\r\n\r\n    schema.sections.forEach((section, si) => {\r\n      section.fields.forEach((field, fi) => {\r\n        // Chiavi duplicate\r\n        if (allKeys.has(field.key)) {\r\n          errors.push({\r\n            type: 'duplicate_key',\r\n            message: `Chiave duplicata: ${field.key}`,\r\n            path: `sections[${si}].fields[${fi}].key`,\r\n          });\r\n        }\r\n        allKeys.add(field.key);\r\n\r\n        // Label mancante\r\n        if (!field.label?.trim()) {\r\n          warnings.push({\r\n            type: 'missing_label',\r\n            message: `Campo senza label: ${field.key}`,\r\n            path: `sections[${si}].fields[${fi}].label`,\r\n          });\r\n        }\r\n\r\n        // Opzioni vuote per campi di scelta\r\n        if (['select', 'radio', 'multiselect'].includes(field.type)) {\r\n          if (!field.options || (Array.isArray(field.options) && field.options.length === 0)) {\r\n            warnings.push({\r\n              type: 'empty_options',\r\n              message: `Campo ${field.key} di tipo ${field.type} senza opzioni`,\r\n              path: `sections[${si}].fields[${fi}].options`,\r\n            });\r\n          }\r\n        }\r\n\r\n        // Riferimenti nelle condizioni\r\n        if (field.conditions) {\r\n          field.conditions.forEach((cond, ci) => {\r\n            if (cond.field && !allKeys.has(cond.field)) {\r\n              errors.push({\r\n                type: 'invalid_reference',\r\n                message: `Riferimento a campo inesistente: ${cond.field} in ${field.key}`,\r\n                path: `sections[${si}].fields[${fi}].conditions[${ci}].field`,\r\n              });\r\n            }\r\n          });\r\n        }\r\n      });\r\n    });\r\n\r\n    return { valid: errors.length === 0, errors, warnings };\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/form-builder-editor
|
|
3
|
+
* Factory per la creazione di field descriptor con valori di default.
|
|
4
|
+
*/
|
|
5
|
+
import { Injectable } from '@angular/core';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* Servizio factory per creare field descriptor con configurazioni
|
|
9
|
+
* di default appropriate per ogni tipo di campo.
|
|
10
|
+
*/
|
|
11
|
+
export class UiEditorFieldFactoryService {
|
|
12
|
+
constructor() {
|
|
13
|
+
/** @internal Contatore per generare chiavi univoche. */
|
|
14
|
+
this.fieldCounter = 0;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Crea un field descriptor di default per il tipo specificato.
|
|
18
|
+
*
|
|
19
|
+
* @param type - Tipo di campo da creare
|
|
20
|
+
* @param key - Chiave opzionale (se omessa viene generata automaticamente)
|
|
21
|
+
* @returns Field descriptor con valori di default
|
|
22
|
+
*/
|
|
23
|
+
createDefaultField(type, key) {
|
|
24
|
+
const generatedKey = key || this.generateUniqueKey(type);
|
|
25
|
+
const baseField = {
|
|
26
|
+
key: generatedKey,
|
|
27
|
+
type,
|
|
28
|
+
label: this.getDefaultLabel(type),
|
|
29
|
+
placeholder: this.getDefaultPlaceholder(type),
|
|
30
|
+
required: false,
|
|
31
|
+
disabled: false,
|
|
32
|
+
readonly: false,
|
|
33
|
+
};
|
|
34
|
+
switch (type) {
|
|
35
|
+
case 'select':
|
|
36
|
+
case 'radio':
|
|
37
|
+
return {
|
|
38
|
+
...baseField,
|
|
39
|
+
options: [
|
|
40
|
+
{ value: 'option1', label: 'Opzione 1' },
|
|
41
|
+
{ value: 'option2', label: 'Opzione 2' },
|
|
42
|
+
],
|
|
43
|
+
hideEmptyOption: false,
|
|
44
|
+
};
|
|
45
|
+
case 'multiselect':
|
|
46
|
+
case 'freemultiselect':
|
|
47
|
+
return {
|
|
48
|
+
...baseField,
|
|
49
|
+
options: [
|
|
50
|
+
{ value: 'option1', label: 'Opzione 1' },
|
|
51
|
+
{ value: 'option2', label: 'Opzione 2' },
|
|
52
|
+
{ value: 'option3', label: 'Opzione 3' },
|
|
53
|
+
],
|
|
54
|
+
};
|
|
55
|
+
case 'checkbox':
|
|
56
|
+
case 'switch':
|
|
57
|
+
return {
|
|
58
|
+
...baseField,
|
|
59
|
+
defaultValue: false,
|
|
60
|
+
appearance: {
|
|
61
|
+
style: type === 'switch' ? 'switch' : 'default',
|
|
62
|
+
color: 'primary',
|
|
63
|
+
},
|
|
64
|
+
};
|
|
65
|
+
case 'number':
|
|
66
|
+
return {
|
|
67
|
+
...baseField,
|
|
68
|
+
defaultValue: 0,
|
|
69
|
+
validation: [{ type: 'min', value: 0, message: 'Il valore deve essere maggiore o uguale a 0' }],
|
|
70
|
+
};
|
|
71
|
+
case 'email':
|
|
72
|
+
return {
|
|
73
|
+
...baseField,
|
|
74
|
+
validation: [{ type: 'email', message: 'Inserire un indirizzo email valido' }],
|
|
75
|
+
};
|
|
76
|
+
case 'date':
|
|
77
|
+
case 'datetime':
|
|
78
|
+
return {
|
|
79
|
+
...baseField,
|
|
80
|
+
placeholder: type === 'date' ? 'gg/mm/aaaa' : 'gg/mm/aaaa hh:mm',
|
|
81
|
+
};
|
|
82
|
+
case 'file':
|
|
83
|
+
return {
|
|
84
|
+
...baseField,
|
|
85
|
+
fileConfig: {
|
|
86
|
+
maxFileSize: 5242880,
|
|
87
|
+
multiple: false,
|
|
88
|
+
showPreview: true,
|
|
89
|
+
enableDragDrop: true,
|
|
90
|
+
acceptedTypes: ['image/*', 'application/pdf'],
|
|
91
|
+
},
|
|
92
|
+
};
|
|
93
|
+
case 'textarea':
|
|
94
|
+
return {
|
|
95
|
+
...baseField,
|
|
96
|
+
validation: [{ type: 'maxLength', value: 500, message: 'Massimo 500 caratteri' }],
|
|
97
|
+
};
|
|
98
|
+
case 'password':
|
|
99
|
+
return {
|
|
100
|
+
...baseField,
|
|
101
|
+
validation: [{ type: 'minLength', value: 8, message: 'La password deve contenere almeno 8 caratteri' }],
|
|
102
|
+
};
|
|
103
|
+
case 'text':
|
|
104
|
+
return {
|
|
105
|
+
...baseField,
|
|
106
|
+
validation: [{ type: 'maxLength', value: 255, message: 'Massimo 255 caratteri' }],
|
|
107
|
+
};
|
|
108
|
+
case 'custom':
|
|
109
|
+
return {
|
|
110
|
+
...baseField,
|
|
111
|
+
customConfig: {
|
|
112
|
+
component: 'custom-component-selector',
|
|
113
|
+
inputs: {},
|
|
114
|
+
config: {},
|
|
115
|
+
},
|
|
116
|
+
};
|
|
117
|
+
case 'divider':
|
|
118
|
+
return { ...baseField, label: 'Separatore' };
|
|
119
|
+
case 'flag':
|
|
120
|
+
return { ...baseField, defaultValue: false };
|
|
121
|
+
default:
|
|
122
|
+
return baseField;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
/** Duplica un campo generando una nuova chiave univoca. */
|
|
126
|
+
duplicateField(field) {
|
|
127
|
+
const newKey = this.generateUniqueKey(field.type);
|
|
128
|
+
return {
|
|
129
|
+
...JSON.parse(JSON.stringify(field)),
|
|
130
|
+
key: newKey,
|
|
131
|
+
label: `${field.label} (Copia)`,
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
// ─── Private ─────────────────────────────────────────────────
|
|
135
|
+
/** @internal Genera una chiave univoca per un campo. */
|
|
136
|
+
generateUniqueKey(type) {
|
|
137
|
+
this.fieldCounter++;
|
|
138
|
+
return `${type}_${this.fieldCounter}_${Date.now()}`;
|
|
139
|
+
}
|
|
140
|
+
/** @internal Mappa tipo -> label di default. */
|
|
141
|
+
getDefaultLabel(type) {
|
|
142
|
+
const labels = {
|
|
143
|
+
text: 'Campo Testo',
|
|
144
|
+
email: 'Email',
|
|
145
|
+
password: 'Password', // NOSONAR - UI label for password field type, not a credential
|
|
146
|
+
number: 'Numero',
|
|
147
|
+
textarea: 'Area Testo',
|
|
148
|
+
select: 'Selezione',
|
|
149
|
+
multiselect: 'Selezione Multipla',
|
|
150
|
+
freemultiselect: 'Selezione Multipla Libera',
|
|
151
|
+
checkbox: 'Checkbox',
|
|
152
|
+
switch: 'Switch',
|
|
153
|
+
radio: 'Radio Button',
|
|
154
|
+
date: 'Data',
|
|
155
|
+
datetime: 'Data e Ora',
|
|
156
|
+
file: 'File',
|
|
157
|
+
custom: 'Campo Personalizzato',
|
|
158
|
+
divider: 'Separatore',
|
|
159
|
+
flag: 'Flag',
|
|
160
|
+
location: 'Localita',
|
|
161
|
+
'location-table': 'Tabella Localita',
|
|
162
|
+
};
|
|
163
|
+
return labels[type] || 'Campo';
|
|
164
|
+
}
|
|
165
|
+
/** @internal Mappa tipo -> placeholder di default. */
|
|
166
|
+
getDefaultPlaceholder(type) {
|
|
167
|
+
const placeholders = {
|
|
168
|
+
text: 'Inserisci il testo',
|
|
169
|
+
email: 'nome@esempio.it',
|
|
170
|
+
password: 'Inserisci la password', // NOSONAR - UI placeholder text for password field, not a credential
|
|
171
|
+
number: 'Inserisci un numero',
|
|
172
|
+
textarea: 'Inserisci il testo...',
|
|
173
|
+
select: "Seleziona un'opzione",
|
|
174
|
+
multiselect: 'Seleziona una o piu opzioni',
|
|
175
|
+
freemultiselect: 'Seleziona o inserisci valori',
|
|
176
|
+
date: 'gg/mm/aaaa',
|
|
177
|
+
datetime: 'gg/mm/aaaa hh:mm',
|
|
178
|
+
file: 'Seleziona un file',
|
|
179
|
+
};
|
|
180
|
+
return placeholders[type] || '';
|
|
181
|
+
}
|
|
182
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFieldFactoryService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
183
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFieldFactoryService }); }
|
|
184
|
+
}
|
|
185
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiEditorFieldFactoryService, decorators: [{
|
|
186
|
+
type: Injectable
|
|
187
|
+
}] });
|
|
188
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"field-factory.service.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/form-builder-editor/services/field-factory.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * @module ng-ui-system/form-builder-editor\r\n * Factory per la creazione di field descriptor con valori di default.\r\n */\r\n\r\nimport { Injectable } from '@angular/core';\r\nimport { UiFormFieldDescriptor, UiFieldType } from '../../form-builder/types/index';\r\n\r\n/**\r\n * Servizio factory per creare field descriptor con configurazioni\r\n * di default appropriate per ogni tipo di campo.\r\n */\r\n@Injectable()\r\nexport class UiEditorFieldFactoryService {\r\n  /** @internal Contatore per generare chiavi univoche. */\r\n  private fieldCounter = 0;\r\n\r\n  /**\r\n   * Crea un field descriptor di default per il tipo specificato.\r\n   *\r\n   * @param type - Tipo di campo da creare\r\n   * @param key - Chiave opzionale (se omessa viene generata automaticamente)\r\n   * @returns Field descriptor con valori di default\r\n   */\r\n  createDefaultField(type: UiFieldType, key?: string): UiFormFieldDescriptor {\r\n    const generatedKey = key || this.generateUniqueKey(type);\r\n\r\n    const baseField: UiFormFieldDescriptor = {\r\n      key: generatedKey,\r\n      type,\r\n      label: this.getDefaultLabel(type),\r\n      placeholder: this.getDefaultPlaceholder(type),\r\n      required: false,\r\n      disabled: false,\r\n      readonly: false,\r\n    };\r\n\r\n    switch (type) {\r\n      case 'select':\r\n      case 'radio':\r\n        return {\r\n          ...baseField,\r\n          options: [\r\n            { value: 'option1', label: 'Opzione 1' },\r\n            { value: 'option2', label: 'Opzione 2' },\r\n          ],\r\n          hideEmptyOption: false,\r\n        };\r\n\r\n      case 'multiselect':\r\n      case 'freemultiselect':\r\n        return {\r\n          ...baseField,\r\n          options: [\r\n            { value: 'option1', label: 'Opzione 1' },\r\n            { value: 'option2', label: 'Opzione 2' },\r\n            { value: 'option3', label: 'Opzione 3' },\r\n          ],\r\n        };\r\n\r\n      case 'checkbox':\r\n      case 'switch':\r\n        return {\r\n          ...baseField,\r\n          defaultValue: false,\r\n          appearance: {\r\n            style: type === 'switch' ? 'switch' : 'default',\r\n            color: 'primary',\r\n          },\r\n        };\r\n\r\n      case 'number':\r\n        return {\r\n          ...baseField,\r\n          defaultValue: 0,\r\n          validation: [{ type: 'min', value: 0, message: 'Il valore deve essere maggiore o uguale a 0' }],\r\n        };\r\n\r\n      case 'email':\r\n        return {\r\n          ...baseField,\r\n          validation: [{ type: 'email', message: 'Inserire un indirizzo email valido' }],\r\n        };\r\n\r\n      case 'date':\r\n      case 'datetime':\r\n        return {\r\n          ...baseField,\r\n          placeholder: type === 'date' ? 'gg/mm/aaaa' : 'gg/mm/aaaa hh:mm',\r\n        };\r\n\r\n      case 'file':\r\n        return {\r\n          ...baseField,\r\n          fileConfig: {\r\n            maxFileSize: 5242880,\r\n            multiple: false,\r\n            showPreview: true,\r\n            enableDragDrop: true,\r\n            acceptedTypes: ['image/*', 'application/pdf'],\r\n          },\r\n        };\r\n\r\n      case 'textarea':\r\n        return {\r\n          ...baseField,\r\n          validation: [{ type: 'maxLength', value: 500, message: 'Massimo 500 caratteri' }],\r\n        };\r\n\r\n      case 'password':\r\n        return {\r\n          ...baseField,\r\n          validation: [{ type: 'minLength', value: 8, message: 'La password deve contenere almeno 8 caratteri' }],\r\n        };\r\n\r\n      case 'text':\r\n        return {\r\n          ...baseField,\r\n          validation: [{ type: 'maxLength', value: 255, message: 'Massimo 255 caratteri' }],\r\n        };\r\n\r\n      case 'custom':\r\n        return {\r\n          ...baseField,\r\n          customConfig: {\r\n            component: 'custom-component-selector',\r\n            inputs: {},\r\n            config: {},\r\n          },\r\n        };\r\n\r\n      case 'divider':\r\n        return { ...baseField, label: 'Separatore' };\r\n\r\n      case 'flag':\r\n        return { ...baseField, defaultValue: false };\r\n\r\n      default:\r\n        return baseField;\r\n    }\r\n  }\r\n\r\n  /** Duplica un campo generando una nuova chiave univoca. */\r\n  duplicateField(field: UiFormFieldDescriptor): UiFormFieldDescriptor {\r\n    const newKey = this.generateUniqueKey(field.type);\r\n    return {\r\n      ...JSON.parse(JSON.stringify(field)),\r\n      key: newKey,\r\n      label: `${field.label} (Copia)`,\r\n    };\r\n  }\r\n\r\n  // ─── Private ─────────────────────────────────────────────────\r\n\r\n  /** @internal Genera una chiave univoca per un campo. */\r\n  private generateUniqueKey(type: UiFieldType): string {\r\n    this.fieldCounter++;\r\n    return `${type}_${this.fieldCounter}_${Date.now()}`;\r\n  }\r\n\r\n  /** @internal Mappa tipo -> label di default. */\r\n  private getDefaultLabel(type: UiFieldType): string {\r\n    const labels: Record<string, string> = {\r\n      text: 'Campo Testo',\r\n      email: 'Email',\r\n      password: 'Password', // NOSONAR - UI label for password field type, not a credential\r\n      number: 'Numero',\r\n      textarea: 'Area Testo',\r\n      select: 'Selezione',\r\n      multiselect: 'Selezione Multipla',\r\n      freemultiselect: 'Selezione Multipla Libera',\r\n      checkbox: 'Checkbox',\r\n      switch: 'Switch',\r\n      radio: 'Radio Button',\r\n      date: 'Data',\r\n      datetime: 'Data e Ora',\r\n      file: 'File',\r\n      custom: 'Campo Personalizzato',\r\n      divider: 'Separatore',\r\n      flag: 'Flag',\r\n      location: 'Localita',\r\n      'location-table': 'Tabella Localita',\r\n    };\r\n    return labels[type] || 'Campo';\r\n  }\r\n\r\n  /** @internal Mappa tipo -> placeholder di default. */\r\n  private getDefaultPlaceholder(type: UiFieldType): string {\r\n    const placeholders: Record<string, string> = {\r\n      text: 'Inserisci il testo',\r\n      email: 'nome@esempio.it',\r\n      password: 'Inserisci la password', // NOSONAR - UI placeholder text for password field, not a credential\r\n      number: 'Inserisci un numero',\r\n      textarea: 'Inserisci il testo...',\r\n      select: \"Seleziona un'opzione\",\r\n      multiselect: 'Seleziona una o piu opzioni',\r\n      freemultiselect: 'Seleziona o inserisci valori',\r\n      date: 'gg/mm/aaaa',\r\n      datetime: 'gg/mm/aaaa hh:mm',\r\n      file: 'Seleziona un file',\r\n    };\r\n    return placeholders[type] || '';\r\n  }\r\n}\r\n"]}
|