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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (141) hide show
  1. package/esm2022/gnggln-ng-ui-system.mjs +5 -0
  2. package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
  3. package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
  4. package/esm2022/lib/components/accordion/index.mjs +2 -0
  5. package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
  6. package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
  7. package/esm2022/lib/components/base-layout/index.mjs +14 -0
  8. package/esm2022/lib/components/button/button-area.component.mjs +196 -0
  9. package/esm2022/lib/components/button/button.component.mjs +164 -0
  10. package/esm2022/lib/components/button/button.types.mjs +6 -0
  11. package/esm2022/lib/components/button/index.mjs +16 -0
  12. package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
  13. package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
  14. package/esm2022/lib/components/crud-table/index.mjs +16 -0
  15. package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
  16. package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
  17. package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
  18. package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
  19. package/esm2022/lib/components/form-builder/index.mjs +19 -0
  20. package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
  21. package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
  22. package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
  23. package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
  24. package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
  25. package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
  26. package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
  27. package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
  28. package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
  29. package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
  30. package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
  31. package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
  32. package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
  33. package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
  34. package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
  35. package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
  36. package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
  37. package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
  38. package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
  39. package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
  40. package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
  41. package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
  42. package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
  43. package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
  44. package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
  45. package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
  46. package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
  47. package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
  48. package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
  49. package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
  50. package/esm2022/lib/components/layout-builder/index.mjs +18 -0
  51. package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
  52. package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
  53. package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
  54. package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
  55. package/esm2022/lib/components/modal/index.mjs +4 -0
  56. package/esm2022/lib/components/modal/modal.component.mjs +139 -0
  57. package/esm2022/lib/components/modal/modal.service.mjs +194 -0
  58. package/esm2022/lib/components/modal/modal.types.mjs +6 -0
  59. package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
  60. package/esm2022/lib/components/page-header/index.mjs +20 -0
  61. package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
  62. package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
  63. package/esm2022/lib/components/table/index.mjs +2 -0
  64. package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
  65. package/esm2022/lib/components/table/table.types.mjs +6 -0
  66. package/esm2022/lib/core/types/index.mjs +6 -0
  67. package/esm2022/lib/core/utils/index.mjs +53 -0
  68. package/esm2022/lib/sources/location-data.opt.json +8942 -0
  69. package/esm2022/lib/sources/nazioni.opt.json +215 -0
  70. package/esm2022/public-api.mjs +34 -0
  71. package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
  72. package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
  73. package/index.d.ts +5 -0
  74. package/lib/components/accordion/accordion.component.d.ts +118 -0
  75. package/lib/components/accordion/accordion.types.d.ts +62 -0
  76. package/lib/components/accordion/index.d.ts +2 -0
  77. package/lib/components/base-layout/base-layout.component.d.ts +83 -0
  78. package/lib/components/base-layout/base-layout.types.d.ts +26 -0
  79. package/lib/components/base-layout/index.d.ts +13 -0
  80. package/lib/components/button/button-area.component.d.ts +88 -0
  81. package/lib/components/button/button.component.d.ts +55 -0
  82. package/lib/components/button/button.types.d.ts +70 -0
  83. package/lib/components/button/index.d.ts +15 -0
  84. package/lib/components/crud-table/crud-table.component.d.ts +143 -0
  85. package/lib/components/crud-table/crud-table.types.d.ts +207 -0
  86. package/lib/components/crud-table/index.d.ts +15 -0
  87. package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
  88. package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
  89. package/lib/components/form-builder/form-builder.component.d.ts +183 -0
  90. package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
  91. package/lib/components/form-builder/index.d.ts +13 -0
  92. package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
  93. package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
  94. package/lib/components/form-builder/services/location.service.d.ts +83 -0
  95. package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
  96. package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
  97. package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
  98. package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
  99. package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
  100. package/lib/components/form-builder/types/condition.types.d.ts +51 -0
  101. package/lib/components/form-builder/types/field.types.d.ts +288 -0
  102. package/lib/components/form-builder/types/index.d.ts +5 -0
  103. package/lib/components/form-builder/types/schema.types.d.ts +227 -0
  104. package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
  105. package/lib/components/form-builder/types/validation.types.d.ts +174 -0
  106. package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
  107. package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
  108. package/lib/components/form-builder-editor/index.d.ts +15 -0
  109. package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
  110. package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
  111. package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
  112. package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
  113. package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
  114. package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
  115. package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
  116. package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
  117. package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
  118. package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
  119. package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
  120. package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
  121. package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
  122. package/lib/components/layout-builder/index.d.ts +16 -0
  123. package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
  124. package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
  125. package/lib/components/layout-builder/layout.service.d.ts +100 -0
  126. package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
  127. package/lib/components/modal/index.d.ts +4 -0
  128. package/lib/components/modal/modal.component.d.ts +44 -0
  129. package/lib/components/modal/modal.service.d.ts +93 -0
  130. package/lib/components/modal/modal.types.d.ts +110 -0
  131. package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
  132. package/lib/components/page-header/index.d.ts +16 -0
  133. package/lib/components/page-header/page-header.component.d.ts +59 -0
  134. package/lib/components/page-header/page-header.types.d.ts +96 -0
  135. package/lib/components/table/index.d.ts +2 -0
  136. package/lib/components/table/paginated-table.component.d.ts +85 -0
  137. package/lib/components/table/table.types.d.ts +81 -0
  138. package/lib/core/types/index.d.ts +57 -0
  139. package/lib/core/utils/index.d.ts +29 -0
  140. package/package.json +44 -0
  141. package/public-api.d.ts +22 -0
@@ -0,0 +1,789 @@
1
+ import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, ChangeDetectorRef, ViewEncapsulation, inject, } from '@angular/core';
2
+ import { NgTemplateOutlet } from '@angular/common';
3
+ import { MatTooltipModule } from '@angular/material/tooltip';
4
+ import { LucideAngularModule } from 'lucide-angular';
5
+ import { Subject } from 'rxjs';
6
+ import { UiButtonAreaComponent } from '../button/button-area.component';
7
+ import { UiFormBuilderComponent } from '../form-builder/form-builder.component';
8
+ import { UiModalService } from '../modal/modal.service';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "@angular/material/tooltip";
11
+ import * as i2 from "lucide-angular";
12
+ /**
13
+ * Tabella CRUD standalone con form builder integrato.
14
+ *
15
+ * Gestisce il ciclo completo delle operazioni CRUD:
16
+ * inserimento, modifica, duplicazione, eliminazione e
17
+ * visualizzazione in sola lettura. Il form e generato
18
+ * automaticamente dallo schema `UiFormSchema`.
19
+ *
20
+ * Utilizza `UiModalService` per la conferma di eliminazione
21
+ * e `UiFormBuilderComponent` per la gestione del form.
22
+ *
23
+ * @selector ui-crud-table
24
+ *
25
+ * @example
26
+ * ```html
27
+ * <ui-crud-table
28
+ * [data]="items"
29
+ * [columns]="columns"
30
+ * [config]="crudConfig"
31
+ * (dataChange)="onDataChange($event)"
32
+ * (itemAdd)="onAdd($event)"
33
+ * (itemEdit)="onEdit($event)"
34
+ * (itemDelete)="onDelete($event)"
35
+ * />
36
+ * ```
37
+ */
38
+ export class UiCrudTableComponent {
39
+ constructor() {
40
+ this.cdr = inject(ChangeDetectorRef);
41
+ this.modalService = inject(UiModalService);
42
+ this.destroy$ = new Subject();
43
+ // ── Inputs ──────────────────────────────────────────────────────
44
+ /** Array di dati da visualizzare nella tabella. */
45
+ this.data = [];
46
+ /** Definizione delle colonne. */
47
+ this.columns = [];
48
+ /** Mostra l'overlay di caricamento. */
49
+ this.loading = false;
50
+ /** Disabilita tutte le azioni CRUD. */
51
+ this.disabled = false;
52
+ // ── Outputs ─────────────────────────────────────────────────────
53
+ /** Emesso ogni volta che la lista dati cambia (insert/edit/duplicate). */
54
+ this.dataChange = new EventEmitter();
55
+ /** Emesso quando viene inserito un nuovo elemento. */
56
+ this.itemAdd = new EventEmitter();
57
+ /** Emesso quando viene modificato un elemento. */
58
+ this.itemEdit = new EventEmitter();
59
+ /** Emesso quando viene eliminato un elemento (dopo conferma). */
60
+ this.itemDelete = new EventEmitter();
61
+ /** Emesso quando viene duplicato un elemento. */
62
+ this.itemDuplicate = new EventEmitter();
63
+ /** Emesso quando l'utente clicca su una riga in modalita view. */
64
+ this.rowClick = new EventEmitter();
65
+ // ── Stato interno ───────────────────────────────────────────────
66
+ /** @internal Modalita corrente. */
67
+ this.currentMode = 'view';
68
+ /** @internal Elemento in fase di modifica/visualizzazione. */
69
+ this.editingItem = null;
70
+ /** @internal Indice dell'elemento in modifica. */
71
+ this.editingIndex = null;
72
+ /** @internal Dati originali dell'elemento prima della modifica. */
73
+ this.lastOriginalData = null;
74
+ /** @internal Cache per i dati iniziali del form. */
75
+ this._formInitialData = {};
76
+ this._lastCacheKey = null;
77
+ /** @internal Pulsanti per il form. */
78
+ this.formButtons = [];
79
+ /** @internal Pulsante aggiungi. */
80
+ this.addButton = [];
81
+ /** @internal Track function per il loop @for. */
82
+ this.trackByFn = (index, item) => {
83
+ const trackBy = this.config?.trackByProperty || 'id';
84
+ return item[trackBy] ?? index;
85
+ };
86
+ }
87
+ // ── Lifecycle ───────────────────────────────────────────────────
88
+ ngOnInit() {
89
+ this.setupButtons();
90
+ }
91
+ ngOnDestroy() {
92
+ this.destroy$.next();
93
+ this.destroy$.complete();
94
+ }
95
+ // ── Getter calcolati ────────────────────────────────────────────
96
+ /** La tabella e in modalita form (insert/edit/duplicate/readonly). */
97
+ get isFormMode() {
98
+ return this.currentMode !== 'view';
99
+ }
100
+ /** La tabella e in modalita sola lettura. */
101
+ get isReadonlyMode() {
102
+ return this.currentMode === 'readonly';
103
+ }
104
+ /** Dati iniziali calcolati per il form builder. */
105
+ get formInitialData() {
106
+ const cacheKey = this.generateCacheKey();
107
+ if (this._lastCacheKey === cacheKey && this._formInitialData) {
108
+ return this._formInitialData;
109
+ }
110
+ let data = {};
111
+ if (this.currentMode === 'edit' || this.currentMode === 'duplicate' || this.currentMode === 'readonly') {
112
+ data = this.initMapper ? this.initMapper(this.editingItem) : this.editingItem || {};
113
+ }
114
+ this._formInitialData = data;
115
+ this._lastCacheKey = cacheKey;
116
+ return this._formInitialData;
117
+ }
118
+ // ── Permessi ────────────────────────────────────────────────────
119
+ /** Verifica se ci sono azioni disponibili (per mostrare la colonna azioni). */
120
+ hasAnyActions() {
121
+ return (this.config?.allowEdit !== false ||
122
+ this.config?.allowDuplicate === true ||
123
+ this.config?.allowDelete !== false ||
124
+ this.config?.allowView === true);
125
+ }
126
+ /** Verifica se l'elemento puo essere modificato. */
127
+ canEditItem(item) {
128
+ if (this.config?.canEdit) {
129
+ return this.config.canEdit(item, this.currentMode === 'readonly');
130
+ }
131
+ return this.config?.allowEdit !== false;
132
+ }
133
+ /** Verifica se l'elemento puo essere eliminato. */
134
+ canDeleteItem(item) {
135
+ if (this.config?.canDelete) {
136
+ return this.config.canDelete(item, this.currentMode === 'readonly');
137
+ }
138
+ return this.config?.allowDelete !== false;
139
+ }
140
+ /** Verifica se l'elemento puo essere duplicato. */
141
+ canDuplicateItem(item) {
142
+ if (this.config?.canDuplicate) {
143
+ return this.config.canDuplicate(item, this.currentMode === 'readonly');
144
+ }
145
+ return this.config?.allowDuplicate === true;
146
+ }
147
+ /** Verifica se l'elemento puo essere visualizzato in readonly. */
148
+ canViewItem(item) {
149
+ if (this.config?.canView) {
150
+ return this.config.canView(item);
151
+ }
152
+ return this.config?.allowView === true;
153
+ }
154
+ // ── Tooltip ─────────────────────────────────────────────────────
155
+ getEditTooltip(item) {
156
+ if (this.config?.getEditTooltip)
157
+ return this.config.getEditTooltip(item);
158
+ return this.canEditItem(item) ? 'Modifica' : 'Modifica non disponibile';
159
+ }
160
+ getDeleteTooltip(item) {
161
+ if (this.config?.getDeleteTooltip)
162
+ return this.config.getDeleteTooltip(item);
163
+ return this.canDeleteItem(item) ? 'Elimina' : 'Eliminazione non disponibile';
164
+ }
165
+ getDuplicateTooltip(item) {
166
+ if (this.config?.getDuplicateTooltip)
167
+ return this.config.getDuplicateTooltip(item);
168
+ return this.canDuplicateItem(item) ? 'Duplica' : 'Duplicazione non disponibile';
169
+ }
170
+ getViewTooltip(item) {
171
+ if (this.config?.getViewTooltip)
172
+ return this.config.getViewTooltip(item);
173
+ return this.canViewItem(item) ? 'Visualizza' : 'Visualizzazione non disponibile';
174
+ }
175
+ // ── Classi CSS per riga ─────────────────────────────────────────
176
+ getRowClasses(item, index) {
177
+ let classes = 'ui-crud-table__row';
178
+ if (this.config?.getRowClass) {
179
+ const extra = this.config.getRowClass(item, index);
180
+ if (Array.isArray(extra)) {
181
+ classes += ' ' + extra.join(' ');
182
+ }
183
+ else if (extra) {
184
+ classes += ' ' + extra;
185
+ }
186
+ }
187
+ return classes;
188
+ }
189
+ // ── Ellipsis ────────────────────────────────────────────────────
190
+ shouldShowEllipsis(col, value) {
191
+ if (!col.ellipsis || value == null)
192
+ return false;
193
+ return String(value).length > (col.maxLength || 50);
194
+ }
195
+ truncateText(col, value) {
196
+ if (value == null)
197
+ return '\u2014';
198
+ const str = String(value);
199
+ const max = col.maxLength || 50;
200
+ return str.length > max ? str.substring(0, max) + '\u2026' : str;
201
+ }
202
+ getFullText(value) {
203
+ return String(value ?? '\u2014');
204
+ }
205
+ // ── Operazioni CRUD ─────────────────────────────────────────────
206
+ /** Attiva la modalita inserimento. */
207
+ activateInsertMode() {
208
+ this.config?.onBeforeFormOpen?.('insert', null, this.data);
209
+ this.currentMode = 'insert';
210
+ this.editingItem = null;
211
+ this.editingIndex = null;
212
+ this.invalidateFormCache();
213
+ this.setupButtons();
214
+ this.cdr.detectChanges();
215
+ }
216
+ /** Attiva la modalita modifica per un elemento. */
217
+ activateEditMode(item, index) {
218
+ if (!this.canEditItem(item))
219
+ return;
220
+ this.config?.onBeforeFormOpen?.('edit', item, this.data);
221
+ this.currentMode = 'edit';
222
+ this.lastOriginalData = item;
223
+ this.editingItem = { ...item };
224
+ this.editingIndex = index;
225
+ this.invalidateFormCache();
226
+ this.setupButtons();
227
+ this.cdr.detectChanges();
228
+ }
229
+ /** Attiva la modalita visualizzazione readonly per un elemento. */
230
+ activateViewMode(item, index) {
231
+ if (!this.canViewItem(item))
232
+ return;
233
+ this.currentMode = 'readonly';
234
+ this.lastOriginalData = item;
235
+ this.editingItem = { ...item };
236
+ this.editingIndex = index;
237
+ this.invalidateFormCache();
238
+ this.setupButtons();
239
+ this.cdr.detectChanges();
240
+ }
241
+ /** Attiva la modalita duplicazione per un elemento. */
242
+ activateDuplicateMode(item) {
243
+ if (!this.canDuplicateItem(item))
244
+ return;
245
+ this.config?.onBeforeFormOpen?.('duplicate', item, this.data);
246
+ this.currentMode = 'duplicate';
247
+ this.editingItem = { ...item };
248
+ this.editingIndex = null;
249
+ this.invalidateFormCache();
250
+ this.setupButtons();
251
+ this.cdr.detectChanges();
252
+ }
253
+ /** Annulla il form e torna alla vista tabella. */
254
+ cancelForm() {
255
+ this.currentMode = 'view';
256
+ this.editingItem = null;
257
+ this.editingIndex = null;
258
+ this.lastOriginalData = null;
259
+ this.invalidateFormCache();
260
+ this.setupButtons();
261
+ this.cdr.detectChanges();
262
+ }
263
+ /** Salva i dati del form (insert/edit/duplicate). */
264
+ saveForm() {
265
+ if (!this.formBuilder)
266
+ return;
267
+ if (!this.formBuilder.isFormValid()) {
268
+ this.formBuilder.validateAllFields();
269
+ return;
270
+ }
271
+ const formData = this.formBuilder.getFormValue();
272
+ const newData = [...this.data];
273
+ switch (this.currentMode) {
274
+ case 'insert':
275
+ newData.push(formData);
276
+ this.itemAdd.emit(formData);
277
+ break;
278
+ case 'duplicate':
279
+ newData.push(formData);
280
+ this.itemDuplicate.emit(formData);
281
+ break;
282
+ case 'edit':
283
+ if (this.editingIndex !== null) {
284
+ newData[this.editingIndex] = formData;
285
+ this.itemEdit.emit({
286
+ item: formData,
287
+ index: this.editingIndex,
288
+ originalData: this.lastOriginalData,
289
+ });
290
+ }
291
+ break;
292
+ }
293
+ this.data = newData;
294
+ this.dataChange.emit(this.data);
295
+ this.cancelForm();
296
+ }
297
+ /** Elimina un elemento con conferma tramite UiModalService. */
298
+ deleteItem(item, index) {
299
+ if (!this.canDeleteItem(item))
300
+ return;
301
+ const message = this.config?.deleteConfirmMessage || 'Sei sicuro di voler eliminare questo elemento?';
302
+ this.modalService
303
+ .confirm({
304
+ title: 'Conferma eliminazione',
305
+ message,
306
+ confirmText: 'Elimina',
307
+ cancelText: 'Annulla',
308
+ variant: 'delete',
309
+ })
310
+ .subscribe((confirmed) => {
311
+ if (confirmed) {
312
+ this.itemDelete.emit({ item, index });
313
+ }
314
+ });
315
+ }
316
+ // ── Event handlers template ─────────────────────────────────────
317
+ onRowClick(item) {
318
+ if (this.currentMode === 'view') {
319
+ this.rowClick.emit(item);
320
+ }
321
+ }
322
+ /** @internal */
323
+ onViewBtnClick(event, item, index) {
324
+ event.stopPropagation();
325
+ if (this.disabled || !this.canViewItem(item))
326
+ return;
327
+ this.activateViewMode(item, index);
328
+ }
329
+ /** @internal */
330
+ onEditBtnClick(event, item, index) {
331
+ event.stopPropagation();
332
+ if (this.disabled || !this.canEditItem(item))
333
+ return;
334
+ this.activateEditMode(item, index);
335
+ }
336
+ /** @internal */
337
+ onDuplicateBtnClick(event, item) {
338
+ event.stopPropagation();
339
+ if (this.disabled || !this.canDuplicateItem(item))
340
+ return;
341
+ this.activateDuplicateMode(item);
342
+ }
343
+ /** @internal */
344
+ onDeleteBtnClick(event, item, index) {
345
+ event.stopPropagation();
346
+ if (this.disabled || !this.canDeleteItem(item))
347
+ return;
348
+ this.deleteItem(item, index);
349
+ }
350
+ // ── Utilita private ─────────────────────────────────────────────
351
+ /** @internal Genera una chiave per la cache dei dati del form. */
352
+ generateCacheKey() {
353
+ if (this.currentMode === 'view' || this.currentMode === 'insert') {
354
+ return this.currentMode;
355
+ }
356
+ const itemId = this.editingItem ? JSON.stringify(this.editingItem) : 'null';
357
+ return `${this.currentMode}-${itemId}`;
358
+ }
359
+ /** @internal Invalida la cache dei dati del form. */
360
+ invalidateFormCache() {
361
+ this._formInitialData = {};
362
+ this._lastCacheKey = null;
363
+ }
364
+ /** @internal Configura i pulsanti in base alla modalita corrente. */
365
+ setupButtons() {
366
+ // Pulsante "Aggiungi" (visibile solo in modalita view)
367
+ this.addButton =
368
+ this.currentMode === 'view' && this.config?.allowAdd !== false && !this.disabled
369
+ ? [
370
+ {
371
+ id: 'crud-add',
372
+ label: this.config?.addButtonLabel || 'Aggiungi',
373
+ variant: 'outline',
374
+ icon: 'plus',
375
+ action: () => this.activateInsertMode(),
376
+ },
377
+ ]
378
+ : [];
379
+ // Pulsanti form (visibili in modalita form)
380
+ this.formButtons = this.isFormMode
381
+ ? [
382
+ {
383
+ id: 'crud-cancel',
384
+ label: this.isReadonlyMode ? 'Chiudi' : this.config?.cancelButtonLabel || 'Annulla',
385
+ variant: 'outline',
386
+ action: () => this.cancelForm(),
387
+ },
388
+ ...(this.isReadonlyMode
389
+ ? []
390
+ : [
391
+ {
392
+ id: 'crud-save',
393
+ label: this.getSaveButtonLabel(),
394
+ variant: 'primary',
395
+ icon: 'check',
396
+ action: () => this.saveForm(),
397
+ },
398
+ ]),
399
+ ]
400
+ : [];
401
+ }
402
+ /** @internal Restituisce la label del pulsante salva in base alla modalita. */
403
+ getSaveButtonLabel() {
404
+ switch (this.currentMode) {
405
+ case 'edit':
406
+ return this.config?.updateButtonLabel || 'Aggiorna';
407
+ case 'duplicate':
408
+ return this.config?.duplicateButtonLabel || 'Duplica';
409
+ default:
410
+ return this.config?.saveButtonLabel || 'Salva';
411
+ }
412
+ }
413
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiCrudTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
414
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiCrudTableComponent, isStandalone: true, selector: "ui-crud-table", inputs: { data: "data", columns: "columns", config: "config", loading: "loading", disabled: "disabled", initMapper: "initMapper" }, outputs: { dataChange: "dataChange", itemAdd: "itemAdd", itemEdit: "itemEdit", itemDelete: "itemDelete", itemDuplicate: "itemDuplicate", rowClick: "rowClick" }, host: { classAttribute: "ui-crud-table-host" }, viewQueries: [{ propertyName: "formBuilder", first: true, predicate: UiFormBuilderComponent, descendants: true }], ngImport: i0, template: `
415
+ <div class="ui-crud-table">
416
+
417
+ <!-- ═══ Stato vuoto ═══════════════════════════════════════ -->
418
+ @if (currentMode === 'view' && (!data || data.length === 0) && !loading) {
419
+ <div class="ui-crud-table__empty">
420
+ <div class="ui-crud-table__empty-message">
421
+ <lucide-icon name="info" [size]="20" class="ui-crud-table__empty-icon" aria-hidden="true" />
422
+ <span>{{ config?.emptyMessage || 'Nessun elemento presente' }}</span>
423
+ </div>
424
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
425
+ </div>
426
+ }
427
+
428
+ <!-- ═══ Stato di caricamento ══════════════════════════════ -->
429
+ @if (loading && currentMode === 'view') {
430
+ <div class="ui-crud-table__loading" role="status" aria-live="polite">
431
+ <span class="ui-crud-table__spinner" aria-hidden="true"></span>
432
+ <span class="ui-crud-table__sr-only">Caricamento in corso</span>
433
+ </div>
434
+ }
435
+
436
+ <!-- ═══ Vista tabella ═════════════════════════════════════ -->
437
+ @if (currentMode === 'view' && data && data.length > 0 && !loading) {
438
+ <div class="ui-crud-table__container">
439
+ <table class="ui-crud-table__table" role="grid" [attr.aria-label]="config?.tableLabel || 'Tabella CRUD'">
440
+ <thead>
441
+ <tr>
442
+ @for (col of columns; track col.key) {
443
+ <th
444
+ class="ui-crud-table__th"
445
+ [style.width]="col.width || null"
446
+ [style.text-align]="col.align || 'left'"
447
+ >
448
+ @if (col.headerTemplate) {
449
+ <ng-container
450
+ [ngTemplateOutlet]="col.headerTemplate"
451
+ [ngTemplateOutletContext]="{ $implicit: col, column: col }"
452
+ />
453
+ } @else {
454
+ {{ col.header }}
455
+ }
456
+ </th>
457
+ }
458
+ @if (!disabled && hasAnyActions()) {
459
+ <th class="ui-crud-table__th ui-crud-table__th--actions">
460
+ Azioni
461
+ </th>
462
+ }
463
+ </tr>
464
+ </thead>
465
+ <tbody>
466
+ @for (item of data; track trackByFn($index, item); let i = $index) {
467
+ <tr
468
+ [class]="getRowClasses(item, i)"
469
+ (click)="onRowClick(item)"
470
+ >
471
+ @for (col of columns; track col.key) {
472
+ <td
473
+ class="ui-crud-table__td"
474
+ [style.text-align]="col.align || 'left'"
475
+ >
476
+ @if (col.template) {
477
+ <ng-container
478
+ [ngTemplateOutlet]="col.template"
479
+ [ngTemplateOutletContext]="{
480
+ $implicit: item,
481
+ row: item,
482
+ column: col,
483
+ index: i,
484
+ value: item[col.key]
485
+ }"
486
+ />
487
+ } @else if (col.ellipsis && shouldShowEllipsis(col, item[col.key])) {
488
+ <span
489
+ class="ui-crud-table__ellipsis"
490
+ [matTooltip]="getFullText(item[col.key])"
491
+ >{{ truncateText(col, item[col.key]) }}</span>
492
+ } @else {
493
+ <span class="ui-crud-table__cell-text">{{ item[col.key] ?? '\u2014' }}</span>
494
+ }
495
+ </td>
496
+ }
497
+
498
+ <!-- Colonna azioni -->
499
+ @if (!disabled && hasAnyActions()) {
500
+ <td class="ui-crud-table__td ui-crud-table__td--actions">
501
+ <div class="ui-crud-table__actions">
502
+ @if (config?.allowView === true) {
503
+ <button
504
+ class="ui-crud-table__action-btn"
505
+ [class.ui-crud-table__action-btn--disabled]="!canViewItem(item)"
506
+ [matTooltip]="getViewTooltip(item)"
507
+ (click)="onViewBtnClick($event, item, i)"
508
+ type="button"
509
+ >
510
+ <lucide-icon name="eye" [size]="16" aria-hidden="true" />
511
+ </button>
512
+ }
513
+ @if (config?.allowEdit !== false) {
514
+ <button
515
+ class="ui-crud-table__action-btn"
516
+ [class.ui-crud-table__action-btn--disabled]="!canEditItem(item)"
517
+ [matTooltip]="getEditTooltip(item)"
518
+ (click)="onEditBtnClick($event, item, i)"
519
+ type="button"
520
+ >
521
+ <lucide-icon name="pencil" [size]="16" aria-hidden="true" />
522
+ </button>
523
+ }
524
+ @if (config?.allowDuplicate === true) {
525
+ <button
526
+ class="ui-crud-table__action-btn"
527
+ [class.ui-crud-table__action-btn--disabled]="!canDuplicateItem(item)"
528
+ [matTooltip]="getDuplicateTooltip(item)"
529
+ (click)="onDuplicateBtnClick($event, item)"
530
+ type="button"
531
+ >
532
+ <lucide-icon name="copy" [size]="16" aria-hidden="true" />
533
+ </button>
534
+ }
535
+ @if (config?.allowDelete !== false) {
536
+ <button
537
+ class="ui-crud-table__action-btn ui-crud-table__action-btn--warn"
538
+ [class.ui-crud-table__action-btn--disabled]="!canDeleteItem(item)"
539
+ [matTooltip]="getDeleteTooltip(item)"
540
+ (click)="onDeleteBtnClick($event, item, i)"
541
+ type="button"
542
+ >
543
+ <lucide-icon name="trash-2" [size]="16" aria-hidden="true" />
544
+ </button>
545
+ }
546
+ </div>
547
+ </td>
548
+ }
549
+ </tr>
550
+ }
551
+ </tbody>
552
+ </table>
553
+ </div>
554
+
555
+ <!-- Footer con pulsante aggiungi -->
556
+ <div class="ui-crud-table__footer">
557
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
558
+ </div>
559
+ }
560
+
561
+ <!-- ═══ Vista form (insert/edit/duplicate/readonly) ══════ -->
562
+ @if (isFormMode) {
563
+ <div class="ui-crud-table__form">
564
+ <div class="ui-crud-table__form-header">
565
+ <h4 class="ui-crud-table__form-title">
566
+ @switch (currentMode) {
567
+ @case ('insert') { Nuovo elemento }
568
+ @case ('edit') { Modifica elemento }
569
+ @case ('duplicate') { Duplica elemento }
570
+ @case ('readonly') { Visualizza elemento }
571
+ }
572
+ </h4>
573
+ </div>
574
+ <div class="ui-crud-table__form-content">
575
+ <ui-form-builder
576
+ [schema]="config.formSchema"
577
+ [initialData]="formInitialData"
578
+ [readonly]="isReadonlyMode"
579
+ [buttonsOverride]="formButtons"
580
+ />
581
+ </div>
582
+ </div>
583
+ }
584
+ </div>
585
+ `, isInline: true, styles: [".ui-crud-table-host{display:block;width:100%}.ui-crud-table{display:flex;flex-direction:column;width:100%;font-family:var(--ui-font-family)}.ui-crud-table__empty{display:flex;justify-content:space-between;align-items:center;padding:var(--ui-spacing-5);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);min-height:80px}.ui-crud-table__empty-message{display:flex;align-items:center;gap:var(--ui-spacing-2);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm);font-style:italic}.ui-crud-table__empty-icon{color:var(--ui-color-text-muted);flex-shrink:0}.ui-crud-table__loading{display:flex;align-items:center;justify-content:center;min-height:120px;padding:var(--ui-spacing-6);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle)}.ui-crud-table__spinner{width:24px;height:24px;border:3px solid var(--ui-color-primary-light);border-top-color:var(--ui-color-primary);border-radius:50%;animation:ui-crud-table-spin .7s linear infinite}@keyframes ui-crud-table-spin{to{transform:rotate(360deg)}}.ui-crud-table__sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-crud-table__container{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);overflow:hidden;overflow-x:auto;box-shadow:var(--ui-shadow-sm)}.ui-crud-table__table{width:100%;border-collapse:collapse;font-size:var(--ui-font-size-sm)}.ui-crud-table__th{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:2px solid var(--ui-color-border);background:var(--ui-color-bg-subtle);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-secondary);text-align:left;white-space:nowrap;-webkit-user-select:none;user-select:none;vertical-align:middle}.ui-crud-table__th--actions{text-align:right;width:140px;min-width:140px}.ui-crud-table__row{transition-property:background-color;transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__row:hover{background:var(--ui-color-surface-hover)}.ui-crud-table__row:last-child .ui-crud-table__td{border-bottom:none}.ui-crud-table__td{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);color:var(--ui-color-text);vertical-align:middle;word-break:break-word}.ui-crud-table__td--actions{text-align:right;white-space:nowrap}.ui-crud-table__cell-text{color:var(--ui-color-text)}.ui-crud-table__ellipsis{cursor:help}.ui-crud-table__ellipsis:hover{text-decoration:underline;text-decoration-style:dotted;text-underline-offset:2px}.ui-crud-table__actions{display:inline-flex;gap:var(--ui-spacing-1);align-items:center;justify-content:flex-end}.ui-crud-table__action-btn{appearance:none;border:none;background:transparent;cursor:pointer;display:inline-flex;align-items:center;justify-content:center;width:32px;height:32px;border-radius:var(--ui-radius-md);color:var(--ui-color-primary);transition-property:\"background-color, color, transform, opacity\";transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__action-btn:focus{outline:none}.ui-crud-table__action-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-crud-table__action-btn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-primary-light);transform:scale(1.08)}.ui-crud-table__action-btn--warn{color:var(--ui-color-warn)}.ui-crud-table__action-btn--warn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-warn-light)}.ui-crud-table__action-btn--disabled{opacity:.35;cursor:not-allowed;color:var(--ui-color-text-muted)!important}.ui-crud-table__action-btn--disabled:hover{transform:none;background:transparent}.ui-crud-table__footer{margin-top:var(--ui-spacing-4)}.ui-crud-table__form{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);box-shadow:var(--ui-shadow-sm);overflow:hidden}.ui-crud-table__form-header{padding:var(--ui-spacing-4) var(--ui-spacing-5);border-bottom:1px solid var(--ui-color-border);background:var(--ui-color-surface)}.ui-crud-table__form-title{margin:0;font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text)}.ui-crud-table__form-content{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-crud-table__th,.ui-crud-table__td{padding:var(--ui-spacing-2) var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-crud-table__th--actions{width:auto;min-width:auto}.ui-crud-table__empty{flex-direction:column;gap:var(--ui-spacing-3);text-align:center}.ui-crud-table__form-content{padding:var(--ui-spacing-3)}}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonAreaComponent, selector: "ui-button-area", inputs: ["buttons", "align", "ariaLabel", "gap", "stackOnMobile", "disableWhileLoading", "loadingIds"] }, { kind: "component", type: UiFormBuilderComponent, selector: "ui-form-builder", inputs: ["schema", "initialData", "readonly", "disabled", "buttonsOverride", "loadingFor"], outputs: ["valueChange", "validationChange", "formSubmit", "formReset", "customEvent"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
586
+ }
587
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiCrudTableComponent, decorators: [{
588
+ type: Component,
589
+ args: [{ selector: 'ui-crud-table', standalone: true, imports: [NgTemplateOutlet, MatTooltipModule, LucideAngularModule, UiButtonAreaComponent, UiFormBuilderComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: { class: 'ui-crud-table-host' }, template: `
590
+ <div class="ui-crud-table">
591
+
592
+ <!-- ═══ Stato vuoto ═══════════════════════════════════════ -->
593
+ @if (currentMode === 'view' && (!data || data.length === 0) && !loading) {
594
+ <div class="ui-crud-table__empty">
595
+ <div class="ui-crud-table__empty-message">
596
+ <lucide-icon name="info" [size]="20" class="ui-crud-table__empty-icon" aria-hidden="true" />
597
+ <span>{{ config?.emptyMessage || 'Nessun elemento presente' }}</span>
598
+ </div>
599
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
600
+ </div>
601
+ }
602
+
603
+ <!-- ═══ Stato di caricamento ══════════════════════════════ -->
604
+ @if (loading && currentMode === 'view') {
605
+ <div class="ui-crud-table__loading" role="status" aria-live="polite">
606
+ <span class="ui-crud-table__spinner" aria-hidden="true"></span>
607
+ <span class="ui-crud-table__sr-only">Caricamento in corso</span>
608
+ </div>
609
+ }
610
+
611
+ <!-- ═══ Vista tabella ═════════════════════════════════════ -->
612
+ @if (currentMode === 'view' && data && data.length > 0 && !loading) {
613
+ <div class="ui-crud-table__container">
614
+ <table class="ui-crud-table__table" role="grid" [attr.aria-label]="config?.tableLabel || 'Tabella CRUD'">
615
+ <thead>
616
+ <tr>
617
+ @for (col of columns; track col.key) {
618
+ <th
619
+ class="ui-crud-table__th"
620
+ [style.width]="col.width || null"
621
+ [style.text-align]="col.align || 'left'"
622
+ >
623
+ @if (col.headerTemplate) {
624
+ <ng-container
625
+ [ngTemplateOutlet]="col.headerTemplate"
626
+ [ngTemplateOutletContext]="{ $implicit: col, column: col }"
627
+ />
628
+ } @else {
629
+ {{ col.header }}
630
+ }
631
+ </th>
632
+ }
633
+ @if (!disabled && hasAnyActions()) {
634
+ <th class="ui-crud-table__th ui-crud-table__th--actions">
635
+ Azioni
636
+ </th>
637
+ }
638
+ </tr>
639
+ </thead>
640
+ <tbody>
641
+ @for (item of data; track trackByFn($index, item); let i = $index) {
642
+ <tr
643
+ [class]="getRowClasses(item, i)"
644
+ (click)="onRowClick(item)"
645
+ >
646
+ @for (col of columns; track col.key) {
647
+ <td
648
+ class="ui-crud-table__td"
649
+ [style.text-align]="col.align || 'left'"
650
+ >
651
+ @if (col.template) {
652
+ <ng-container
653
+ [ngTemplateOutlet]="col.template"
654
+ [ngTemplateOutletContext]="{
655
+ $implicit: item,
656
+ row: item,
657
+ column: col,
658
+ index: i,
659
+ value: item[col.key]
660
+ }"
661
+ />
662
+ } @else if (col.ellipsis && shouldShowEllipsis(col, item[col.key])) {
663
+ <span
664
+ class="ui-crud-table__ellipsis"
665
+ [matTooltip]="getFullText(item[col.key])"
666
+ >{{ truncateText(col, item[col.key]) }}</span>
667
+ } @else {
668
+ <span class="ui-crud-table__cell-text">{{ item[col.key] ?? '\u2014' }}</span>
669
+ }
670
+ </td>
671
+ }
672
+
673
+ <!-- Colonna azioni -->
674
+ @if (!disabled && hasAnyActions()) {
675
+ <td class="ui-crud-table__td ui-crud-table__td--actions">
676
+ <div class="ui-crud-table__actions">
677
+ @if (config?.allowView === true) {
678
+ <button
679
+ class="ui-crud-table__action-btn"
680
+ [class.ui-crud-table__action-btn--disabled]="!canViewItem(item)"
681
+ [matTooltip]="getViewTooltip(item)"
682
+ (click)="onViewBtnClick($event, item, i)"
683
+ type="button"
684
+ >
685
+ <lucide-icon name="eye" [size]="16" aria-hidden="true" />
686
+ </button>
687
+ }
688
+ @if (config?.allowEdit !== false) {
689
+ <button
690
+ class="ui-crud-table__action-btn"
691
+ [class.ui-crud-table__action-btn--disabled]="!canEditItem(item)"
692
+ [matTooltip]="getEditTooltip(item)"
693
+ (click)="onEditBtnClick($event, item, i)"
694
+ type="button"
695
+ >
696
+ <lucide-icon name="pencil" [size]="16" aria-hidden="true" />
697
+ </button>
698
+ }
699
+ @if (config?.allowDuplicate === true) {
700
+ <button
701
+ class="ui-crud-table__action-btn"
702
+ [class.ui-crud-table__action-btn--disabled]="!canDuplicateItem(item)"
703
+ [matTooltip]="getDuplicateTooltip(item)"
704
+ (click)="onDuplicateBtnClick($event, item)"
705
+ type="button"
706
+ >
707
+ <lucide-icon name="copy" [size]="16" aria-hidden="true" />
708
+ </button>
709
+ }
710
+ @if (config?.allowDelete !== false) {
711
+ <button
712
+ class="ui-crud-table__action-btn ui-crud-table__action-btn--warn"
713
+ [class.ui-crud-table__action-btn--disabled]="!canDeleteItem(item)"
714
+ [matTooltip]="getDeleteTooltip(item)"
715
+ (click)="onDeleteBtnClick($event, item, i)"
716
+ type="button"
717
+ >
718
+ <lucide-icon name="trash-2" [size]="16" aria-hidden="true" />
719
+ </button>
720
+ }
721
+ </div>
722
+ </td>
723
+ }
724
+ </tr>
725
+ }
726
+ </tbody>
727
+ </table>
728
+ </div>
729
+
730
+ <!-- Footer con pulsante aggiungi -->
731
+ <div class="ui-crud-table__footer">
732
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
733
+ </div>
734
+ }
735
+
736
+ <!-- ═══ Vista form (insert/edit/duplicate/readonly) ══════ -->
737
+ @if (isFormMode) {
738
+ <div class="ui-crud-table__form">
739
+ <div class="ui-crud-table__form-header">
740
+ <h4 class="ui-crud-table__form-title">
741
+ @switch (currentMode) {
742
+ @case ('insert') { Nuovo elemento }
743
+ @case ('edit') { Modifica elemento }
744
+ @case ('duplicate') { Duplica elemento }
745
+ @case ('readonly') { Visualizza elemento }
746
+ }
747
+ </h4>
748
+ </div>
749
+ <div class="ui-crud-table__form-content">
750
+ <ui-form-builder
751
+ [schema]="config.formSchema"
752
+ [initialData]="formInitialData"
753
+ [readonly]="isReadonlyMode"
754
+ [buttonsOverride]="formButtons"
755
+ />
756
+ </div>
757
+ </div>
758
+ }
759
+ </div>
760
+ `, styles: [".ui-crud-table-host{display:block;width:100%}.ui-crud-table{display:flex;flex-direction:column;width:100%;font-family:var(--ui-font-family)}.ui-crud-table__empty{display:flex;justify-content:space-between;align-items:center;padding:var(--ui-spacing-5);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);min-height:80px}.ui-crud-table__empty-message{display:flex;align-items:center;gap:var(--ui-spacing-2);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm);font-style:italic}.ui-crud-table__empty-icon{color:var(--ui-color-text-muted);flex-shrink:0}.ui-crud-table__loading{display:flex;align-items:center;justify-content:center;min-height:120px;padding:var(--ui-spacing-6);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle)}.ui-crud-table__spinner{width:24px;height:24px;border:3px solid var(--ui-color-primary-light);border-top-color:var(--ui-color-primary);border-radius:50%;animation:ui-crud-table-spin .7s linear infinite}@keyframes ui-crud-table-spin{to{transform:rotate(360deg)}}.ui-crud-table__sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-crud-table__container{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);overflow:hidden;overflow-x:auto;box-shadow:var(--ui-shadow-sm)}.ui-crud-table__table{width:100%;border-collapse:collapse;font-size:var(--ui-font-size-sm)}.ui-crud-table__th{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:2px solid var(--ui-color-border);background:var(--ui-color-bg-subtle);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-secondary);text-align:left;white-space:nowrap;-webkit-user-select:none;user-select:none;vertical-align:middle}.ui-crud-table__th--actions{text-align:right;width:140px;min-width:140px}.ui-crud-table__row{transition-property:background-color;transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__row:hover{background:var(--ui-color-surface-hover)}.ui-crud-table__row:last-child .ui-crud-table__td{border-bottom:none}.ui-crud-table__td{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);color:var(--ui-color-text);vertical-align:middle;word-break:break-word}.ui-crud-table__td--actions{text-align:right;white-space:nowrap}.ui-crud-table__cell-text{color:var(--ui-color-text)}.ui-crud-table__ellipsis{cursor:help}.ui-crud-table__ellipsis:hover{text-decoration:underline;text-decoration-style:dotted;text-underline-offset:2px}.ui-crud-table__actions{display:inline-flex;gap:var(--ui-spacing-1);align-items:center;justify-content:flex-end}.ui-crud-table__action-btn{appearance:none;border:none;background:transparent;cursor:pointer;display:inline-flex;align-items:center;justify-content:center;width:32px;height:32px;border-radius:var(--ui-radius-md);color:var(--ui-color-primary);transition-property:\"background-color, color, transform, opacity\";transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__action-btn:focus{outline:none}.ui-crud-table__action-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-crud-table__action-btn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-primary-light);transform:scale(1.08)}.ui-crud-table__action-btn--warn{color:var(--ui-color-warn)}.ui-crud-table__action-btn--warn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-warn-light)}.ui-crud-table__action-btn--disabled{opacity:.35;cursor:not-allowed;color:var(--ui-color-text-muted)!important}.ui-crud-table__action-btn--disabled:hover{transform:none;background:transparent}.ui-crud-table__footer{margin-top:var(--ui-spacing-4)}.ui-crud-table__form{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);box-shadow:var(--ui-shadow-sm);overflow:hidden}.ui-crud-table__form-header{padding:var(--ui-spacing-4) var(--ui-spacing-5);border-bottom:1px solid var(--ui-color-border);background:var(--ui-color-surface)}.ui-crud-table__form-title{margin:0;font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text)}.ui-crud-table__form-content{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-crud-table__th,.ui-crud-table__td{padding:var(--ui-spacing-2) var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-crud-table__th--actions{width:auto;min-width:auto}.ui-crud-table__empty{flex-direction:column;gap:var(--ui-spacing-3);text-align:center}.ui-crud-table__form-content{padding:var(--ui-spacing-3)}}\n"] }]
761
+ }], propDecorators: { data: [{
762
+ type: Input
763
+ }], columns: [{
764
+ type: Input
765
+ }], config: [{
766
+ type: Input
767
+ }], loading: [{
768
+ type: Input
769
+ }], disabled: [{
770
+ type: Input
771
+ }], initMapper: [{
772
+ type: Input
773
+ }], dataChange: [{
774
+ type: Output
775
+ }], itemAdd: [{
776
+ type: Output
777
+ }], itemEdit: [{
778
+ type: Output
779
+ }], itemDelete: [{
780
+ type: Output
781
+ }], itemDuplicate: [{
782
+ type: Output
783
+ }], rowClick: [{
784
+ type: Output
785
+ }], formBuilder: [{
786
+ type: ViewChild,
787
+ args: [UiFormBuilderComponent]
788
+ }] } });
789
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-table.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/crud-table/crud-table.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  ViewChild,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  ViewEncapsulation,\r\n  OnInit,\r\n  OnDestroy,\r\n  inject,\r\n} from '@angular/core';\r\nimport { NgTemplateOutlet } from '@angular/common';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subject } from 'rxjs';\r\n\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiButtonDescriptor } from '../button/button.types';\r\nimport { UiFormBuilderComponent } from '../form-builder/form-builder.component';\r\nimport { UiModalService } from '../modal/modal.service';\r\nimport { UiFormData } from '../form-builder/types/schema.types';\r\n\r\nimport {\r\n  UiCrudTableColumn,\r\n  UiCrudTableConfig,\r\n  UiCrudMode,\r\n  UiCrudEditEvent,\r\n  UiCrudDeleteEvent,\r\n} from './crud-table.types';\r\n\r\n/**\r\n * Tabella CRUD standalone con form builder integrato.\r\n *\r\n * Gestisce il ciclo completo delle operazioni CRUD:\r\n * inserimento, modifica, duplicazione, eliminazione e\r\n * visualizzazione in sola lettura. Il form e generato\r\n * automaticamente dallo schema `UiFormSchema`.\r\n *\r\n * Utilizza `UiModalService` per la conferma di eliminazione\r\n * e `UiFormBuilderComponent` per la gestione del form.\r\n *\r\n * @selector ui-crud-table\r\n *\r\n * @example\r\n * ```html\r\n * <ui-crud-table\r\n *   [data]=\"items\"\r\n *   [columns]=\"columns\"\r\n *   [config]=\"crudConfig\"\r\n *   (dataChange)=\"onDataChange($event)\"\r\n *   (itemAdd)=\"onAdd($event)\"\r\n *   (itemEdit)=\"onEdit($event)\"\r\n *   (itemDelete)=\"onDelete($event)\"\r\n * />\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-crud-table',\r\n  standalone: true,\r\n  imports: [NgTemplateOutlet, MatTooltipModule, LucideAngularModule, UiButtonAreaComponent, UiFormBuilderComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: { class: 'ui-crud-table-host' },\r\n  template: `\r\n    <div class=\"ui-crud-table\">\r\n\r\n      <!-- ═══ Stato vuoto ═══════════════════════════════════════ -->\r\n      @if (currentMode === 'view' && (!data || data.length === 0) && !loading) {\r\n        <div class=\"ui-crud-table__empty\">\r\n          <div class=\"ui-crud-table__empty-message\">\r\n            <lucide-icon name=\"info\" [size]=\"20\" class=\"ui-crud-table__empty-icon\" aria-hidden=\"true\" />\r\n            <span>{{ config?.emptyMessage || 'Nessun elemento presente' }}</span>\r\n          </div>\r\n          <ui-button-area [buttons]=\"addButton\" align=\"end\" gap=\"xs\" />\r\n        </div>\r\n      }\r\n\r\n      <!-- ═══ Stato di caricamento ══════════════════════════════ -->\r\n      @if (loading && currentMode === 'view') {\r\n        <div class=\"ui-crud-table__loading\" role=\"status\" aria-live=\"polite\">\r\n          <span class=\"ui-crud-table__spinner\" aria-hidden=\"true\"></span>\r\n          <span class=\"ui-crud-table__sr-only\">Caricamento in corso</span>\r\n        </div>\r\n      }\r\n\r\n      <!-- ═══ Vista tabella ═════════════════════════════════════ -->\r\n      @if (currentMode === 'view' && data && data.length > 0 && !loading) {\r\n        <div class=\"ui-crud-table__container\">\r\n          <table class=\"ui-crud-table__table\" role=\"grid\" [attr.aria-label]=\"config?.tableLabel || 'Tabella CRUD'\">\r\n            <thead>\r\n              <tr>\r\n                @for (col of columns; track col.key) {\r\n                  <th\r\n                    class=\"ui-crud-table__th\"\r\n                    [style.width]=\"col.width || null\"\r\n                    [style.text-align]=\"col.align || 'left'\"\r\n                  >\r\n                    @if (col.headerTemplate) {\r\n                      <ng-container\r\n                        [ngTemplateOutlet]=\"col.headerTemplate\"\r\n                        [ngTemplateOutletContext]=\"{ $implicit: col, column: col }\"\r\n                      />\r\n                    } @else {\r\n                      {{ col.header }}\r\n                    }\r\n                  </th>\r\n                }\r\n                @if (!disabled && hasAnyActions()) {\r\n                  <th class=\"ui-crud-table__th ui-crud-table__th--actions\">\r\n                    Azioni\r\n                  </th>\r\n                }\r\n              </tr>\r\n            </thead>\r\n            <tbody>\r\n              @for (item of data; track trackByFn($index, item); let i = $index) {\r\n                <tr\r\n                  [class]=\"getRowClasses(item, i)\"\r\n                  (click)=\"onRowClick(item)\"\r\n                >\r\n                  @for (col of columns; track col.key) {\r\n                    <td\r\n                      class=\"ui-crud-table__td\"\r\n                      [style.text-align]=\"col.align || 'left'\"\r\n                    >\r\n                      @if (col.template) {\r\n                        <ng-container\r\n                          [ngTemplateOutlet]=\"col.template\"\r\n                          [ngTemplateOutletContext]=\"{\r\n                            $implicit: item,\r\n                            row: item,\r\n                            column: col,\r\n                            index: i,\r\n                            value: item[col.key]\r\n                          }\"\r\n                        />\r\n                      } @else if (col.ellipsis && shouldShowEllipsis(col, item[col.key])) {\r\n                        <span\r\n                          class=\"ui-crud-table__ellipsis\"\r\n                          [matTooltip]=\"getFullText(item[col.key])\"\r\n                        >{{ truncateText(col, item[col.key]) }}</span>\r\n                      } @else {\r\n                        <span class=\"ui-crud-table__cell-text\">{{ item[col.key] ?? '\\u2014' }}</span>\r\n                      }\r\n                    </td>\r\n                  }\r\n\r\n                  <!-- Colonna azioni -->\r\n                  @if (!disabled && hasAnyActions()) {\r\n                    <td class=\"ui-crud-table__td ui-crud-table__td--actions\">\r\n                      <div class=\"ui-crud-table__actions\">\r\n                        @if (config?.allowView === true) {\r\n                          <button\r\n                            class=\"ui-crud-table__action-btn\"\r\n                            [class.ui-crud-table__action-btn--disabled]=\"!canViewItem(item)\"\r\n                            [matTooltip]=\"getViewTooltip(item)\"\r\n                            (click)=\"onViewBtnClick($event, item, i)\"\r\n                            type=\"button\"\r\n                          >\r\n                            <lucide-icon name=\"eye\" [size]=\"16\" aria-hidden=\"true\" />\r\n                          </button>\r\n                        }\r\n                        @if (config?.allowEdit !== false) {\r\n                          <button\r\n                            class=\"ui-crud-table__action-btn\"\r\n                            [class.ui-crud-table__action-btn--disabled]=\"!canEditItem(item)\"\r\n                            [matTooltip]=\"getEditTooltip(item)\"\r\n                            (click)=\"onEditBtnClick($event, item, i)\"\r\n                            type=\"button\"\r\n                          >\r\n                            <lucide-icon name=\"pencil\" [size]=\"16\" aria-hidden=\"true\" />\r\n                          </button>\r\n                        }\r\n                        @if (config?.allowDuplicate === true) {\r\n                          <button\r\n                            class=\"ui-crud-table__action-btn\"\r\n                            [class.ui-crud-table__action-btn--disabled]=\"!canDuplicateItem(item)\"\r\n                            [matTooltip]=\"getDuplicateTooltip(item)\"\r\n                            (click)=\"onDuplicateBtnClick($event, item)\"\r\n                            type=\"button\"\r\n                          >\r\n                            <lucide-icon name=\"copy\" [size]=\"16\" aria-hidden=\"true\" />\r\n                          </button>\r\n                        }\r\n                        @if (config?.allowDelete !== false) {\r\n                          <button\r\n                            class=\"ui-crud-table__action-btn ui-crud-table__action-btn--warn\"\r\n                            [class.ui-crud-table__action-btn--disabled]=\"!canDeleteItem(item)\"\r\n                            [matTooltip]=\"getDeleteTooltip(item)\"\r\n                            (click)=\"onDeleteBtnClick($event, item, i)\"\r\n                            type=\"button\"\r\n                          >\r\n                            <lucide-icon name=\"trash-2\" [size]=\"16\" aria-hidden=\"true\" />\r\n                          </button>\r\n                        }\r\n                      </div>\r\n                    </td>\r\n                  }\r\n                </tr>\r\n              }\r\n            </tbody>\r\n          </table>\r\n        </div>\r\n\r\n        <!-- Footer con pulsante aggiungi -->\r\n        <div class=\"ui-crud-table__footer\">\r\n          <ui-button-area [buttons]=\"addButton\" align=\"end\" gap=\"xs\" />\r\n        </div>\r\n      }\r\n\r\n      <!-- ═══ Vista form (insert/edit/duplicate/readonly) ══════ -->\r\n      @if (isFormMode) {\r\n        <div class=\"ui-crud-table__form\">\r\n          <div class=\"ui-crud-table__form-header\">\r\n            <h4 class=\"ui-crud-table__form-title\">\r\n              @switch (currentMode) {\r\n                @case ('insert')    { Nuovo elemento }\r\n                @case ('edit')      { Modifica elemento }\r\n                @case ('duplicate') { Duplica elemento }\r\n                @case ('readonly')  { Visualizza elemento }\r\n              }\r\n            </h4>\r\n          </div>\r\n          <div class=\"ui-crud-table__form-content\">\r\n            <ui-form-builder\r\n              [schema]=\"config.formSchema\"\r\n              [initialData]=\"formInitialData\"\r\n              [readonly]=\"isReadonlyMode\"\r\n              [buttonsOverride]=\"formButtons\"\r\n            />\r\n          </div>\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './crud-table.component.scss',\r\n})\r\nexport class UiCrudTableComponent<T extends Record<string, any> = Record<string, any>> implements OnInit, OnDestroy {\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n  private readonly modalService = inject(UiModalService);\r\n  private readonly destroy$ = new Subject<void>();\r\n\r\n  // ── Inputs ──────────────────────────────────────────────────────\r\n\r\n  /** Array di dati da visualizzare nella tabella. */\r\n  @Input() data: T[] = [];\r\n\r\n  /** Definizione delle colonne. */\r\n  @Input() columns: UiCrudTableColumn[] = [];\r\n\r\n  /** Configurazione CRUD (schema form, permessi, label). */\r\n  @Input() config!: UiCrudTableConfig;\r\n\r\n  /** Mostra l'overlay di caricamento. */\r\n  @Input() loading = false;\r\n\r\n  /** Disabilita tutte le azioni CRUD. */\r\n  @Input() disabled = false;\r\n\r\n  /**\r\n   * Funzione di mappatura applicata ai dati di un elemento\r\n   * prima di passarli come `initialData` al form builder.\r\n   * Utile per trasformare la struttura dati dell'oggetto.\r\n   */\r\n  @Input() initMapper?: (originalData: T) => Record<string, any>;\r\n\r\n  // ── Outputs ─────────────────────────────────────────────────────\r\n\r\n  /** Emesso ogni volta che la lista dati cambia (insert/edit/duplicate). */\r\n  @Output() dataChange = new EventEmitter<T[]>();\r\n\r\n  /** Emesso quando viene inserito un nuovo elemento. */\r\n  @Output() itemAdd = new EventEmitter<T>();\r\n\r\n  /** Emesso quando viene modificato un elemento. */\r\n  @Output() itemEdit = new EventEmitter<UiCrudEditEvent<T>>();\r\n\r\n  /** Emesso quando viene eliminato un elemento (dopo conferma). */\r\n  @Output() itemDelete = new EventEmitter<UiCrudDeleteEvent<T>>();\r\n\r\n  /** Emesso quando viene duplicato un elemento. */\r\n  @Output() itemDuplicate = new EventEmitter<T>();\r\n\r\n  /** Emesso quando l'utente clicca su una riga in modalita view. */\r\n  @Output() rowClick = new EventEmitter<T>();\r\n\r\n  // ── Stato interno ───────────────────────────────────────────────\r\n\r\n  /** @internal Modalita corrente. */\r\n  currentMode: UiCrudMode = 'view';\r\n\r\n  /** @internal Elemento in fase di modifica/visualizzazione. */\r\n  editingItem: T | null = null;\r\n\r\n  /** @internal Indice dell'elemento in modifica. */\r\n  editingIndex: number | null = null;\r\n\r\n  /** @internal Dati originali dell'elemento prima della modifica. */\r\n  private lastOriginalData: any = null;\r\n\r\n  /** @internal Cache per i dati iniziali del form. */\r\n  private _formInitialData: UiFormData = {};\r\n  private _lastCacheKey: string | null = null;\r\n\r\n  /** @internal Riferimento al form builder. */\r\n  @ViewChild(UiFormBuilderComponent) formBuilder!: UiFormBuilderComponent;\r\n\r\n  /** @internal Pulsanti per il form. */\r\n  formButtons: UiButtonDescriptor[] = [];\r\n\r\n  /** @internal Pulsante aggiungi. */\r\n  addButton: UiButtonDescriptor[] = [];\r\n\r\n  // ── Lifecycle ───────────────────────────────────────────────────\r\n\r\n  ngOnInit(): void {\r\n    this.setupButtons();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  // ── Getter calcolati ────────────────────────────────────────────\r\n\r\n  /** La tabella e in modalita form (insert/edit/duplicate/readonly). */\r\n  get isFormMode(): boolean {\r\n    return this.currentMode !== 'view';\r\n  }\r\n\r\n  /** La tabella e in modalita sola lettura. */\r\n  get isReadonlyMode(): boolean {\r\n    return this.currentMode === 'readonly';\r\n  }\r\n\r\n  /** Dati iniziali calcolati per il form builder. */\r\n  get formInitialData(): UiFormData {\r\n    const cacheKey = this.generateCacheKey();\r\n    if (this._lastCacheKey === cacheKey && this._formInitialData) {\r\n      return this._formInitialData;\r\n    }\r\n\r\n    let data: UiFormData = {};\r\n    if (this.currentMode === 'edit' || this.currentMode === 'duplicate' || this.currentMode === 'readonly') {\r\n      data = this.initMapper ? this.initMapper(this.editingItem as T) : (this.editingItem as UiFormData) || {};\r\n    }\r\n\r\n    this._formInitialData = data;\r\n    this._lastCacheKey = cacheKey;\r\n    return this._formInitialData;\r\n  }\r\n\r\n  // ── Permessi ────────────────────────────────────────────────────\r\n\r\n  /** Verifica se ci sono azioni disponibili (per mostrare la colonna azioni). */\r\n  hasAnyActions(): boolean {\r\n    return (\r\n      this.config?.allowEdit !== false ||\r\n      this.config?.allowDuplicate === true ||\r\n      this.config?.allowDelete !== false ||\r\n      this.config?.allowView === true\r\n    );\r\n  }\r\n\r\n  /** Verifica se l'elemento puo essere modificato. */\r\n  canEditItem(item: any): boolean {\r\n    if (this.config?.canEdit) {\r\n      return this.config.canEdit(item, this.currentMode === 'readonly');\r\n    }\r\n    return this.config?.allowEdit !== false;\r\n  }\r\n\r\n  /** Verifica se l'elemento puo essere eliminato. */\r\n  canDeleteItem(item: any): boolean {\r\n    if (this.config?.canDelete) {\r\n      return this.config.canDelete(item, this.currentMode === 'readonly');\r\n    }\r\n    return this.config?.allowDelete !== false;\r\n  }\r\n\r\n  /** Verifica se l'elemento puo essere duplicato. */\r\n  canDuplicateItem(item: any): boolean {\r\n    if (this.config?.canDuplicate) {\r\n      return this.config.canDuplicate(item, this.currentMode === 'readonly');\r\n    }\r\n    return this.config?.allowDuplicate === true;\r\n  }\r\n\r\n  /** Verifica se l'elemento puo essere visualizzato in readonly. */\r\n  canViewItem(item: any): boolean {\r\n    if (this.config?.canView) {\r\n      return this.config.canView(item);\r\n    }\r\n    return this.config?.allowView === true;\r\n  }\r\n\r\n  // ── Tooltip ─────────────────────────────────────────────────────\r\n\r\n  getEditTooltip(item: any): string {\r\n    if (this.config?.getEditTooltip) return this.config.getEditTooltip(item);\r\n    return this.canEditItem(item) ? 'Modifica' : 'Modifica non disponibile';\r\n  }\r\n\r\n  getDeleteTooltip(item: any): string {\r\n    if (this.config?.getDeleteTooltip) return this.config.getDeleteTooltip(item);\r\n    return this.canDeleteItem(item) ? 'Elimina' : 'Eliminazione non disponibile';\r\n  }\r\n\r\n  getDuplicateTooltip(item: any): string {\r\n    if (this.config?.getDuplicateTooltip) return this.config.getDuplicateTooltip(item);\r\n    return this.canDuplicateItem(item) ? 'Duplica' : 'Duplicazione non disponibile';\r\n  }\r\n\r\n  getViewTooltip(item: any): string {\r\n    if (this.config?.getViewTooltip) return this.config.getViewTooltip(item);\r\n    return this.canViewItem(item) ? 'Visualizza' : 'Visualizzazione non disponibile';\r\n  }\r\n\r\n  // ── Classi CSS per riga ─────────────────────────────────────────\r\n\r\n  getRowClasses(item: any, index: number): string {\r\n    let classes = 'ui-crud-table__row';\r\n    if (this.config?.getRowClass) {\r\n      const extra = this.config.getRowClass(item, index);\r\n      if (Array.isArray(extra)) {\r\n        classes += ' ' + extra.join(' ');\r\n      } else if (extra) {\r\n        classes += ' ' + extra;\r\n      }\r\n    }\r\n    return classes;\r\n  }\r\n\r\n  // ── Ellipsis ────────────────────────────────────────────────────\r\n\r\n  shouldShowEllipsis(col: UiCrudTableColumn, value: any): boolean {\r\n    if (!col.ellipsis || value == null) return false;\r\n    return String(value).length > (col.maxLength || 50);\r\n  }\r\n\r\n  truncateText(col: UiCrudTableColumn, value: any): string {\r\n    if (value == null) return '\\u2014';\r\n    const str = String(value);\r\n    const max = col.maxLength || 50;\r\n    return str.length > max ? str.substring(0, max) + '\\u2026' : str;\r\n  }\r\n\r\n  getFullText(value: any): string {\r\n    return String(value ?? '\\u2014');\r\n  }\r\n\r\n  // ── Operazioni CRUD ─────────────────────────────────────────────\r\n\r\n  /** Attiva la modalita inserimento. */\r\n  activateInsertMode(): void {\r\n    this.config?.onBeforeFormOpen?.('insert', null, this.data);\r\n    this.currentMode = 'insert';\r\n    this.editingItem = null;\r\n    this.editingIndex = null;\r\n    this.invalidateFormCache();\r\n    this.setupButtons();\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /** Attiva la modalita modifica per un elemento. */\r\n  activateEditMode(item: T, index: number): void {\r\n    if (!this.canEditItem(item)) return;\r\n    this.config?.onBeforeFormOpen?.('edit', item, this.data);\r\n    this.currentMode = 'edit';\r\n    this.lastOriginalData = item;\r\n    this.editingItem = { ...item };\r\n    this.editingIndex = index;\r\n    this.invalidateFormCache();\r\n    this.setupButtons();\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /** Attiva la modalita visualizzazione readonly per un elemento. */\r\n  activateViewMode(item: T, index: number): void {\r\n    if (!this.canViewItem(item)) return;\r\n    this.currentMode = 'readonly';\r\n    this.lastOriginalData = item;\r\n    this.editingItem = { ...item };\r\n    this.editingIndex = index;\r\n    this.invalidateFormCache();\r\n    this.setupButtons();\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /** Attiva la modalita duplicazione per un elemento. */\r\n  activateDuplicateMode(item: T): void {\r\n    if (!this.canDuplicateItem(item)) return;\r\n    this.config?.onBeforeFormOpen?.('duplicate', item, this.data);\r\n    this.currentMode = 'duplicate';\r\n    this.editingItem = { ...item };\r\n    this.editingIndex = null;\r\n    this.invalidateFormCache();\r\n    this.setupButtons();\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /** Annulla il form e torna alla vista tabella. */\r\n  cancelForm(): void {\r\n    this.currentMode = 'view';\r\n    this.editingItem = null;\r\n    this.editingIndex = null;\r\n    this.lastOriginalData = null;\r\n    this.invalidateFormCache();\r\n    this.setupButtons();\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /** Salva i dati del form (insert/edit/duplicate). */\r\n  saveForm(): void {\r\n    if (!this.formBuilder) return;\r\n    if (!this.formBuilder.isFormValid()) {\r\n      this.formBuilder.validateAllFields();\r\n      return;\r\n    }\r\n\r\n    const formData = this.formBuilder.getFormValue() as T;\r\n    const newData = [...this.data];\r\n\r\n    switch (this.currentMode) {\r\n      case 'insert':\r\n        newData.push(formData);\r\n        this.itemAdd.emit(formData);\r\n        break;\r\n      case 'duplicate':\r\n        newData.push(formData);\r\n        this.itemDuplicate.emit(formData);\r\n        break;\r\n      case 'edit':\r\n        if (this.editingIndex !== null) {\r\n          newData[this.editingIndex] = formData;\r\n          this.itemEdit.emit({\r\n            item: formData,\r\n            index: this.editingIndex,\r\n            originalData: this.lastOriginalData,\r\n          });\r\n        }\r\n        break;\r\n    }\r\n\r\n    this.data = newData;\r\n    this.dataChange.emit(this.data);\r\n    this.cancelForm();\r\n  }\r\n\r\n  /** Elimina un elemento con conferma tramite UiModalService. */\r\n  deleteItem(item: T, index: number): void {\r\n    if (!this.canDeleteItem(item)) return;\r\n\r\n    const message = this.config?.deleteConfirmMessage || 'Sei sicuro di voler eliminare questo elemento?';\r\n\r\n    this.modalService\r\n      .confirm({\r\n        title: 'Conferma eliminazione',\r\n        message,\r\n        confirmText: 'Elimina',\r\n        cancelText: 'Annulla',\r\n        variant: 'delete',\r\n      })\r\n      .subscribe((confirmed) => {\r\n        if (confirmed) {\r\n          this.itemDelete.emit({ item, index });\r\n        }\r\n      });\r\n  }\r\n\r\n  // ── Event handlers template ─────────────────────────────────────\r\n\r\n  onRowClick(item: T): void {\r\n    if (this.currentMode === 'view') {\r\n      this.rowClick.emit(item);\r\n    }\r\n  }\r\n\r\n  /** @internal */\r\n  onViewBtnClick(event: Event, item: T, index: number): void {\r\n    event.stopPropagation();\r\n    if (this.disabled || !this.canViewItem(item)) return;\r\n    this.activateViewMode(item, index);\r\n  }\r\n\r\n  /** @internal */\r\n  onEditBtnClick(event: Event, item: T, index: number): void {\r\n    event.stopPropagation();\r\n    if (this.disabled || !this.canEditItem(item)) return;\r\n    this.activateEditMode(item, index);\r\n  }\r\n\r\n  /** @internal */\r\n  onDuplicateBtnClick(event: Event, item: T): void {\r\n    event.stopPropagation();\r\n    if (this.disabled || !this.canDuplicateItem(item)) return;\r\n    this.activateDuplicateMode(item);\r\n  }\r\n\r\n  /** @internal */\r\n  onDeleteBtnClick(event: Event, item: T, index: number): void {\r\n    event.stopPropagation();\r\n    if (this.disabled || !this.canDeleteItem(item)) return;\r\n    this.deleteItem(item, index);\r\n  }\r\n\r\n  /** @internal Track function per il loop @for. */\r\n  trackByFn = (index: number, item: any): any => {\r\n    const trackBy = this.config?.trackByProperty || 'id';\r\n    return item[trackBy] ?? index;\r\n  };\r\n\r\n  // ── Utilita private ─────────────────────────────────────────────\r\n\r\n  /** @internal Genera una chiave per la cache dei dati del form. */\r\n  private generateCacheKey(): string {\r\n    if (this.currentMode === 'view' || this.currentMode === 'insert') {\r\n      return this.currentMode;\r\n    }\r\n    const itemId = this.editingItem ? JSON.stringify(this.editingItem) : 'null';\r\n    return `${this.currentMode}-${itemId}`;\r\n  }\r\n\r\n  /** @internal Invalida la cache dei dati del form. */\r\n  private invalidateFormCache(): void {\r\n    this._formInitialData = {};\r\n    this._lastCacheKey = null;\r\n  }\r\n\r\n  /** @internal Configura i pulsanti in base alla modalita corrente. */\r\n  private setupButtons(): void {\r\n    // Pulsante \"Aggiungi\" (visibile solo in modalita view)\r\n    this.addButton =\r\n      this.currentMode === 'view' && this.config?.allowAdd !== false && !this.disabled\r\n        ? [\r\n            {\r\n              id: 'crud-add',\r\n              label: this.config?.addButtonLabel || 'Aggiungi',\r\n              variant: 'outline',\r\n              icon: 'plus',\r\n              action: () => this.activateInsertMode(),\r\n            },\r\n          ]\r\n        : [];\r\n\r\n    // Pulsanti form (visibili in modalita form)\r\n    this.formButtons = this.isFormMode\r\n      ? [\r\n          {\r\n            id: 'crud-cancel',\r\n            label: this.isReadonlyMode ? 'Chiudi' : this.config?.cancelButtonLabel || 'Annulla',\r\n            variant: 'outline',\r\n            action: () => this.cancelForm(),\r\n          },\r\n          ...(this.isReadonlyMode\r\n            ? []\r\n            : [\r\n                {\r\n                  id: 'crud-save',\r\n                  label: this.getSaveButtonLabel(),\r\n                  variant: 'primary' as const,\r\n                  icon: 'check' as const,\r\n                  action: () => this.saveForm(),\r\n                },\r\n              ]),\r\n        ]\r\n      : [];\r\n  }\r\n\r\n  /** @internal Restituisce la label del pulsante salva in base alla modalita. */\r\n  private getSaveButtonLabel(): string {\r\n    switch (this.currentMode) {\r\n      case 'edit':\r\n        return this.config?.updateButtonLabel || 'Aggiorna';\r\n      case 'duplicate':\r\n        return this.config?.duplicateButtonLabel || 'Duplica';\r\n      default:\r\n        return this.config?.saveButtonLabel || 'Salva';\r\n    }\r\n  }\r\n}\r\n"]}