@gnggln/ng-ui-system 1.0.0-alpha.16 → 1.0.0-alpha.18

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 (92) hide show
  1. package/base-layout/lib/components/base-layout/base-layout.component.d.ts +30 -2
  2. package/base-layout/lib/components/base-layout/base-layout.types.d.ts +14 -0
  3. package/base-layout/lib/components/base-layout/index.d.ts +1 -1
  4. package/base-layout/lib/components/button/button.component.d.ts +8 -2
  5. package/base-layout/lib/components/button/button.types.d.ts +5 -0
  6. package/blackbox/lib/components/button/button.component.d.ts +8 -2
  7. package/blackbox/lib/components/button/button.types.d.ts +5 -0
  8. package/button/lib/components/button/button.component.d.ts +8 -2
  9. package/button/lib/components/button/button.types.d.ts +5 -0
  10. package/crud-table/lib/components/button/button.component.d.ts +8 -2
  11. package/crud-table/lib/components/button/button.types.d.ts +5 -0
  12. package/crud-table/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +9 -6
  13. package/crud-table/lib/components/form-builder/types/schema.types.d.ts +20 -2
  14. package/esm2022/base-layout/lib/components/base-layout/base-layout.component.mjs +71 -16
  15. package/esm2022/base-layout/lib/components/base-layout/base-layout.types.mjs +1 -1
  16. package/esm2022/base-layout/lib/components/base-layout/index.mjs +1 -1
  17. package/esm2022/base-layout/lib/components/button/button-area.component.mjs +3 -1
  18. package/esm2022/base-layout/lib/components/button/button.component.mjs +21 -5
  19. package/esm2022/base-layout/lib/components/button/button.types.mjs +1 -1
  20. package/esm2022/blackbox/lib/components/button/button-area.component.mjs +3 -1
  21. package/esm2022/blackbox/lib/components/button/button.component.mjs +21 -5
  22. package/esm2022/blackbox/lib/components/button/button.types.mjs +1 -1
  23. package/esm2022/button/lib/components/button/button-area.component.mjs +3 -1
  24. package/esm2022/button/lib/components/button/button.component.mjs +21 -5
  25. package/esm2022/button/lib/components/button/button.types.mjs +1 -1
  26. package/esm2022/crud-table/lib/components/button/button-area.component.mjs +3 -1
  27. package/esm2022/crud-table/lib/components/button/button.component.mjs +21 -5
  28. package/esm2022/crud-table/lib/components/button/button.types.mjs +1 -1
  29. package/esm2022/crud-table/lib/components/form-builder/form-builder.component.mjs +3 -3
  30. package/esm2022/crud-table/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +142 -113
  31. package/esm2022/crud-table/lib/components/form-builder/types/schema.types.mjs +1 -1
  32. package/esm2022/form-builder/lib/components/button/button-area.component.mjs +3 -1
  33. package/esm2022/form-builder/lib/components/button/button.component.mjs +21 -5
  34. package/esm2022/form-builder/lib/components/button/button.types.mjs +1 -1
  35. package/esm2022/form-builder/lib/components/form-builder/form-builder.component.mjs +3 -3
  36. package/esm2022/form-builder/lib/components/form-builder/form-wizard.component.mjs +176 -39
  37. package/esm2022/form-builder/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +142 -113
  38. package/esm2022/form-builder/lib/components/form-builder/types/schema.types.mjs +1 -1
  39. package/esm2022/form-builder-editor/lib/components/button/button-area.component.mjs +3 -1
  40. package/esm2022/form-builder-editor/lib/components/button/button.component.mjs +21 -5
  41. package/esm2022/form-builder-editor/lib/components/button/button.types.mjs +1 -1
  42. package/esm2022/form-builder-editor/lib/components/form-builder/form-builder.component.mjs +3 -3
  43. package/esm2022/form-builder-editor/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +142 -113
  44. package/esm2022/form-builder-editor/lib/components/form-builder/types/schema.types.mjs +1 -1
  45. package/esm2022/lib/components/base-layout/base-layout.component.mjs +71 -16
  46. package/esm2022/lib/components/base-layout/base-layout.types.mjs +1 -1
  47. package/esm2022/lib/components/base-layout/index.mjs +1 -1
  48. package/esm2022/lib/components/button/button-area.component.mjs +3 -1
  49. package/esm2022/lib/components/button/button.component.mjs +21 -5
  50. package/esm2022/lib/components/button/button.types.mjs +1 -1
  51. package/esm2022/lib/components/form-builder/form-builder.component.mjs +3 -3
  52. package/esm2022/lib/components/form-builder/form-wizard.component.mjs +176 -39
  53. package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +142 -113
  54. package/esm2022/lib/components/form-builder/types/schema.types.mjs +1 -1
  55. package/esm2022/modal/lib/components/button/button.component.mjs +21 -5
  56. package/esm2022/modal/lib/components/button/button.types.mjs +1 -1
  57. package/fesm2022/gnggln-ng-ui-system-base-layout.mjs +92 -19
  58. package/fesm2022/gnggln-ng-ui-system-base-layout.mjs.map +1 -1
  59. package/fesm2022/gnggln-ng-ui-system-blackbox.mjs +22 -4
  60. package/fesm2022/gnggln-ng-ui-system-blackbox.mjs.map +1 -1
  61. package/fesm2022/gnggln-ng-ui-system-button.mjs +22 -4
  62. package/fesm2022/gnggln-ng-ui-system-button.mjs.map +1 -1
  63. package/fesm2022/gnggln-ng-ui-system-crud-table.mjs +165 -118
  64. package/fesm2022/gnggln-ng-ui-system-crud-table.mjs.map +1 -1
  65. package/fesm2022/gnggln-ng-ui-system-form-builder-editor.mjs +165 -118
  66. package/fesm2022/gnggln-ng-ui-system-form-builder-editor.mjs.map +1 -1
  67. package/fesm2022/gnggln-ng-ui-system-form-builder.mjs +340 -156
  68. package/fesm2022/gnggln-ng-ui-system-form-builder.mjs.map +1 -1
  69. package/fesm2022/gnggln-ng-ui-system-modal.mjs +20 -4
  70. package/fesm2022/gnggln-ng-ui-system-modal.mjs.map +1 -1
  71. package/fesm2022/gnggln-ng-ui-system.mjs +409 -170
  72. package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -1
  73. package/form-builder/lib/components/button/button.component.d.ts +8 -2
  74. package/form-builder/lib/components/button/button.types.d.ts +5 -0
  75. package/form-builder/lib/components/form-builder/form-wizard.component.d.ts +14 -2
  76. package/form-builder/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +9 -6
  77. package/form-builder/lib/components/form-builder/types/schema.types.d.ts +20 -2
  78. package/form-builder-editor/lib/components/button/button.component.d.ts +8 -2
  79. package/form-builder-editor/lib/components/button/button.types.d.ts +5 -0
  80. package/form-builder-editor/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +9 -6
  81. package/form-builder-editor/lib/components/form-builder/types/schema.types.d.ts +20 -2
  82. package/lib/components/base-layout/base-layout.component.d.ts +30 -2
  83. package/lib/components/base-layout/base-layout.types.d.ts +14 -0
  84. package/lib/components/base-layout/index.d.ts +1 -1
  85. package/lib/components/button/button.component.d.ts +8 -2
  86. package/lib/components/button/button.types.d.ts +5 -0
  87. package/lib/components/form-builder/form-wizard.component.d.ts +14 -2
  88. package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +9 -6
  89. package/lib/components/form-builder/types/schema.types.d.ts +20 -2
  90. package/modal/lib/components/button/button.component.d.ts +8 -2
  91. package/modal/lib/components/button/button.types.d.ts +5 -0
  92. package/package.json +18 -18
@@ -2,6 +2,12 @@ import { type OnInit } from '@angular/core';
2
2
  import type { UiButtonAreaAlign, UiButtonDescriptor } from '../button/button.types';
3
3
  import type { UiBackButtonConfig } from './base-layout.types';
4
4
  import * as i0 from "@angular/core";
5
+ /** Stato letto dalla rotta foglia per header/footer opzionali e `data-route-id`. */
6
+ interface UiBaseLayoutRouteSnapshotFlags {
7
+ breadcrumbOff: boolean;
8
+ footerNavOff: boolean;
9
+ routeId: string | null;
10
+ }
5
11
  /**
6
12
  * Full-page layout component that combines a page header (breadcrumbs + title),
7
13
  * a main content area via `<ng-content>`, and a sticky footer action bar
@@ -10,6 +16,14 @@ import * as i0 from "@angular/core";
10
16
  * Optionally renders an automatic "back" button in the footer, derived from
11
17
  * the breadcrumb trail (navigates to the parent breadcrumb).
12
18
  *
19
+ * ### Route `data`
20
+ *
21
+ * Sulla rotta attiva (foglia) puoi impostare:
22
+ *
23
+ * - `breadcrumbOff: true` — nasconde `ui-page-header`
24
+ * - `footerNavOff: true` — nasconde `ui-base-layout__footer`
25
+ * - `id: string` — valorizza `data-route-id` sul contenitore `.ui-base-layout`
26
+ *
13
27
  * @selector ui-base-layout
14
28
  *
15
29
  * @example
@@ -60,6 +74,15 @@ export declare class UiBaseLayoutComponent implements OnInit {
60
74
  private readonly activatedRoute;
61
75
  private readonly breadcrumbService;
62
76
  private readonly destroyRef;
77
+ /**
78
+ * Flag derivati dalla catena attiva (`breadcrumbOff`, `footerNavOff`, `id`).
79
+ * Aggiornati a ogni `NavigationEnd`.
80
+ */
81
+ protected readonly routeSnapshotFlags: import("@angular/core").WritableSignal<UiBaseLayoutRouteSnapshotFlags>;
82
+ /** Header visibile se abilitato da input e non disattivato da `data.breadcrumbOff`. */
83
+ protected readonly effectiveShowHeader: () => boolean;
84
+ /** Footer azioni visibile se ci sono azioni visibili e non disattivato da `data.footerNavOff`. */
85
+ protected readonly effectiveShowFooterNav: () => boolean;
63
86
  /**
64
87
  * Merged action list (back button + consumer actions).
65
88
  * Computed on every change-detection check so it stays in sync
@@ -69,8 +92,12 @@ export declare class UiBaseLayoutComponent implements OnInit {
69
92
  /** Whether the footer should render (at least one non-hidden action exists). */
70
93
  get hasVisibleActions(): boolean;
71
94
  ngOnInit(): void;
72
- /** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */
73
- private refreshBreadcrumbs;
95
+ /** @internal Allinea breadcrumb service e flag layout alla rotta corrente. */
96
+ private syncFromActivatedRoute;
97
+ /** @internal Legge `breadcrumbOff`, `footerNavOff`, `id` dalla catena `pathFromRoot` / foglia. */
98
+ private refreshRouteSnapshotFlags;
99
+ /** @internal Percorre fino al `ActivatedRoute` foglia sotto il punto di iniezione. */
100
+ private deepestChildRoute;
74
101
  /** @internal Determina se il back button deve essere nascosto per la rotta corrente. */
75
102
  private shouldHideBackButton;
76
103
  /** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */
@@ -78,3 +105,4 @@ export declare class UiBaseLayoutComponent implements OnInit {
78
105
  static ɵfac: i0.ɵɵFactoryDeclaration<UiBaseLayoutComponent, never>;
79
106
  static ɵcmp: i0.ɵɵComponentDeclaration<UiBaseLayoutComponent, "ui-base-layout", never, { "showHeader": { "alias": "showHeader"; "required": false; }; "title": { "alias": "title"; "required": false; }; "homeRoute": { "alias": "homeRoute"; "required": false; }; "showHome": { "alias": "showHome"; "required": false; }; "updateDocumentTitle": { "alias": "updateDocumentTitle"; "required": false; }; "titleSuffix": { "alias": "titleSuffix"; "required": false; }; "showBackButton": { "alias": "showBackButton"; "required": false; }; "backButtonConfig": { "alias": "backButtonConfig"; "required": false; }; "actions": { "alias": "actions"; "required": false; }; "footerAlign": { "alias": "footerAlign"; "required": false; }; "footerGap": { "alias": "footerGap"; "required": false; }; }, {}, never, ["*"], true, never>;
80
107
  }
108
+ export {};
@@ -4,6 +4,20 @@
4
4
  */
5
5
  import { UiButtonDescriptor, UiButtonAreaAlign } from '../button/button.types';
6
6
  export { UiButtonDescriptor, UiButtonAreaAlign };
7
+ /**
8
+ * Chiavi opzionali di `data` sulla rotta attiva riconosciute da `UiBaseLayoutComponent`.
9
+ *
10
+ * - `id`: stessa risalita usata dai breadcrumb (dal segmento foglia verso il root: primo `id` trovato).
11
+ * - `breadcrumbOff` / `footerNavOff`: se `true` su **qualsiasi** segmento in `pathFromRoot`, l’area viene nascosta.
12
+ */
13
+ export interface UiBaseLayoutRouteData {
14
+ /** Identificativo stabile della rotta; reso come `data-route-id` sul wrapper `.ui-base-layout`. */
15
+ id?: string;
16
+ /** Se `true`, non viene renderizzato `ui-page-header` (breadcrumb e titolo). */
17
+ breadcrumbOff?: boolean;
18
+ /** Se `true`, non viene renderizzata la barra `ui-base-layout__footer` (azioni / indietro). */
19
+ footerNavOff?: boolean;
20
+ }
7
21
  /**
8
22
  * Configuration for the automatic "back" button rendered in the footer action area.
9
23
  *
@@ -10,4 +10,4 @@
10
10
  * ```
11
11
  */
12
12
  export { UiBaseLayoutComponent } from './base-layout.component';
13
- export { UiBackButtonConfig } from './base-layout.types';
13
+ export { UiBackButtonConfig, UiBaseLayoutRouteData } from './base-layout.types';
@@ -17,8 +17,8 @@ import * as i0 from "@angular/core";
17
17
  * <!-- Loading -->
18
18
  * <ui-button label="Submitting..." variant="primary" [loading]="true" />
19
19
  *
20
- * <!-- Icon-only -->
21
- * <ui-button icon="trash-2" variant="warn" ariaLabel="Delete item" (click)="delete()" />
20
+ * <!-- Icon-only: tooltip o ariaLabel forniscono il nome accessibile -->
21
+ * <ui-button icon="trash-2" variant="warn" tooltip="Elimina" (click)="delete()" />
22
22
  * ```
23
23
  */
24
24
  export declare class UiButtonComponent {
@@ -46,6 +46,12 @@ export declare class UiButtonComponent {
46
46
  ariaLabel?: string;
47
47
  /** Additional CSS class for edge cases. */
48
48
  customClass?: string;
49
+ /**
50
+ * Nome accessibile risolto per il pulsante nativo.
51
+ * Usa `ariaLabel` se impostato; per pulsanti solo icona senza testo visibile
52
+ * ricade sul `tooltip` (il tooltip Material non sostituisce il nome accessibile).
53
+ */
54
+ get effectiveAriaLabel(): string | undefined;
49
55
  /** Computed icon pixel size based on button size. */
50
56
  get iconSize(): number;
51
57
  /** Assembled CSS class string for the native button element. */
@@ -40,6 +40,11 @@ export interface UiButtonDescriptor {
40
40
  label: string;
41
41
  /** Tooltip text shown on hover. */
42
42
  tooltip?: string;
43
+ /**
44
+ * Nome accessibile esplicito (override).
45
+ * Per pulsanti solo icona senza `label`, se omesso si usa `tooltip` come `aria-label`.
46
+ */
47
+ ariaLabel?: string;
43
48
  /**
44
49
  * Click handler. Supports both sync and async callbacks.
45
50
  * Receives the native MouseEvent as argument.
@@ -17,8 +17,8 @@ import * as i0 from "@angular/core";
17
17
  * <!-- Loading -->
18
18
  * <ui-button label="Submitting..." variant="primary" [loading]="true" />
19
19
  *
20
- * <!-- Icon-only -->
21
- * <ui-button icon="trash-2" variant="warn" ariaLabel="Delete item" (click)="delete()" />
20
+ * <!-- Icon-only: tooltip o ariaLabel forniscono il nome accessibile -->
21
+ * <ui-button icon="trash-2" variant="warn" tooltip="Elimina" (click)="delete()" />
22
22
  * ```
23
23
  */
24
24
  export declare class UiButtonComponent {
@@ -46,6 +46,12 @@ export declare class UiButtonComponent {
46
46
  ariaLabel?: string;
47
47
  /** Additional CSS class for edge cases. */
48
48
  customClass?: string;
49
+ /**
50
+ * Nome accessibile risolto per il pulsante nativo.
51
+ * Usa `ariaLabel` se impostato; per pulsanti solo icona senza testo visibile
52
+ * ricade sul `tooltip` (il tooltip Material non sostituisce il nome accessibile).
53
+ */
54
+ get effectiveAriaLabel(): string | undefined;
49
55
  /** Computed icon pixel size based on button size. */
50
56
  get iconSize(): number;
51
57
  /** Assembled CSS class string for the native button element. */
@@ -40,6 +40,11 @@ export interface UiButtonDescriptor {
40
40
  label: string;
41
41
  /** Tooltip text shown on hover. */
42
42
  tooltip?: string;
43
+ /**
44
+ * Nome accessibile esplicito (override).
45
+ * Per pulsanti solo icona senza `label`, se omesso si usa `tooltip` come `aria-label`.
46
+ */
47
+ ariaLabel?: string;
43
48
  /**
44
49
  * Click handler. Supports both sync and async callbacks.
45
50
  * Receives the native MouseEvent as argument.
@@ -17,8 +17,8 @@ import * as i0 from "@angular/core";
17
17
  * <!-- Loading -->
18
18
  * <ui-button label="Submitting..." variant="primary" [loading]="true" />
19
19
  *
20
- * <!-- Icon-only -->
21
- * <ui-button icon="trash-2" variant="warn" ariaLabel="Delete item" (click)="delete()" />
20
+ * <!-- Icon-only: tooltip o ariaLabel forniscono il nome accessibile -->
21
+ * <ui-button icon="trash-2" variant="warn" tooltip="Elimina" (click)="delete()" />
22
22
  * ```
23
23
  */
24
24
  export declare class UiButtonComponent {
@@ -46,6 +46,12 @@ export declare class UiButtonComponent {
46
46
  ariaLabel?: string;
47
47
  /** Additional CSS class for edge cases. */
48
48
  customClass?: string;
49
+ /**
50
+ * Nome accessibile risolto per il pulsante nativo.
51
+ * Usa `ariaLabel` se impostato; per pulsanti solo icona senza testo visibile
52
+ * ricade sul `tooltip` (il tooltip Material non sostituisce il nome accessibile).
53
+ */
54
+ get effectiveAriaLabel(): string | undefined;
49
55
  /** Computed icon pixel size based on button size. */
50
56
  get iconSize(): number;
51
57
  /** Assembled CSS class string for the native button element. */
@@ -40,6 +40,11 @@ export interface UiButtonDescriptor {
40
40
  label: string;
41
41
  /** Tooltip text shown on hover. */
42
42
  tooltip?: string;
43
+ /**
44
+ * Nome accessibile esplicito (override).
45
+ * Per pulsanti solo icona senza `label`, se omesso si usa `tooltip` come `aria-label`.
46
+ */
47
+ ariaLabel?: string;
43
48
  /**
44
49
  * Click handler. Supports both sync and async callbacks.
45
50
  * Receives the native MouseEvent as argument.
@@ -17,8 +17,8 @@ import * as i0 from "@angular/core";
17
17
  * <!-- Loading -->
18
18
  * <ui-button label="Submitting..." variant="primary" [loading]="true" />
19
19
  *
20
- * <!-- Icon-only -->
21
- * <ui-button icon="trash-2" variant="warn" ariaLabel="Delete item" (click)="delete()" />
20
+ * <!-- Icon-only: tooltip o ariaLabel forniscono il nome accessibile -->
21
+ * <ui-button icon="trash-2" variant="warn" tooltip="Elimina" (click)="delete()" />
22
22
  * ```
23
23
  */
24
24
  export declare class UiButtonComponent {
@@ -46,6 +46,12 @@ export declare class UiButtonComponent {
46
46
  ariaLabel?: string;
47
47
  /** Additional CSS class for edge cases. */
48
48
  customClass?: string;
49
+ /**
50
+ * Nome accessibile risolto per il pulsante nativo.
51
+ * Usa `ariaLabel` se impostato; per pulsanti solo icona senza testo visibile
52
+ * ricade sul `tooltip` (il tooltip Material non sostituisce il nome accessibile).
53
+ */
54
+ get effectiveAriaLabel(): string | undefined;
49
55
  /** Computed icon pixel size based on button size. */
50
56
  get iconSize(): number;
51
57
  /** Assembled CSS class string for the native button element. */
@@ -40,6 +40,11 @@ export interface UiButtonDescriptor {
40
40
  label: string;
41
41
  /** Tooltip text shown on hover. */
42
42
  tooltip?: string;
43
+ /**
44
+ * Nome accessibile esplicito (override).
45
+ * Per pulsanti solo icona senza `label`, se omesso si usa `tooltip` come `aria-label`.
46
+ */
47
+ ariaLabel?: string;
43
48
  /**
44
49
  * Click handler. Supports both sync and async callbacks.
45
50
  * Receives the native MouseEvent as argument.
@@ -2,11 +2,10 @@ import { EventEmitter } from '@angular/core';
2
2
  import { UiFormErrorDetail } from '../../types/validation.types';
3
3
  import * as i0 from "@angular/core";
4
4
  /**
5
- * Componente per il riepilogo errori del form.
5
+ * Riepilogo stato validazione del form.
6
6
  *
7
- * Mostra un indicatore con il conteggio degli errori e un dropdown
8
- * con il dettaglio per campo. Click su un errore emette `fieldClick`
9
- * per consentire lo scroll-to-field con highlight nel form builder.
7
+ * Con errori: indicatore + dropdown hover/click per il dettaglio e navigazione ai campi.
8
+ * Senza errori: messaggio di stato positivo fisso (no dropdown), UI prevedibile.
10
9
  *
11
10
  * @selector ui-form-error-summary
12
11
  */
@@ -15,14 +14,18 @@ export declare class UiFormErrorSummaryComponent {
15
14
  errors: UiFormErrorDetail[];
16
15
  /** Conteggio totale degli errori (somma di tutti i messaggi). */
17
16
  totalErrorCount: number;
17
+ /** Testo mostrato quando non ci sono errori. */
18
+ validMessage: string;
18
19
  /** Emesso quando l'utente clicca su un errore per navigare al campo. */
19
20
  fieldClick: EventEmitter<string>;
20
- /** @internal Stato del dropdown. */
21
+ /** @internal Stato del dropdown (solo con errori). */
21
22
  showDropdown: boolean;
23
+ /** True se il form ha almeno un errore da mostrare nel summary. */
24
+ get hasErrors(): boolean;
22
25
  /** Conteggio da visualizzare: usa totalErrorCount se fornito, altrimenti conta dai dettagli. */
23
26
  get displayCount(): number;
24
27
  /** @internal Gestione click su errore. */
25
28
  onFieldClick(fieldKey: string): void;
26
29
  static ɵfac: i0.ɵɵFactoryDeclaration<UiFormErrorSummaryComponent, never>;
27
- static ɵcmp: i0.ɵɵComponentDeclaration<UiFormErrorSummaryComponent, "ui-form-error-summary", never, { "errors": { "alias": "errors"; "required": false; }; "totalErrorCount": { "alias": "totalErrorCount"; "required": false; }; }, { "fieldClick": "fieldClick"; }, never, never, true, never>;
30
+ static ɵcmp: i0.ɵɵComponentDeclaration<UiFormErrorSummaryComponent, "ui-form-error-summary", never, { "errors": { "alias": "errors"; "required": false; }; "totalErrorCount": { "alias": "totalErrorCount"; "required": false; }; "validMessage": { "alias": "validMessage"; "required": false; }; }, { "fieldClick": "fieldClick"; }, never, never, true, never>;
28
31
  }
@@ -147,8 +147,15 @@ export interface UiFormBuilderConfig {
147
147
  columns?: number;
148
148
  /** Mostra i pulsanti di azione (submit/reset). @default true */
149
149
  showButtons?: boolean;
150
- /** Nasconde l'intero footer (pulsanti + error summary). */
150
+ /** Nasconde solo l'area pulsanti del footer (`ui-button-area`), non il summary errori. */
151
151
  hideFooter?: boolean;
152
+ /**
153
+ * Nasconde del tutto il blocco `ui-form-error-summary`.
154
+ * Se assente/falsy il summary è visibile anche senza errori (stato ok prevedibile).
155
+ */
156
+ hideErrorSummary?: boolean;
157
+ /** Messaggio nel summary quando il form non ha errori (stato ok). */
158
+ errorSummaryValidMessage?: string;
152
159
  /** Label personalizzate per i pulsanti. */
153
160
  buttonLabels?: {
154
161
  submit?: string;
@@ -168,7 +175,7 @@ export interface UiFormBuilderConfig {
168
175
  * showProgress: true,
169
176
  * progressStyle: 'steps',
170
177
  * allowBackNavigation: true,
171
- * buttonLabels: { next: 'Avanti', previous: 'Indietro', finish: 'Completa' },
178
+ * buttonLabels: { submit: 'Salva e prosegui', previous: 'Indietro', finish: 'Completa' },
172
179
  * };
173
180
  * ```
174
181
  */
@@ -193,10 +200,21 @@ export interface UiWizardConfig {
193
200
  showNavigationButtons?: boolean;
194
201
  /** Posizione dei pulsanti di navigazione. @default 'bottom' */
195
202
  navigationPosition?: 'top' | 'bottom' | 'both';
203
+ /**
204
+ * Consente di saltare a uno step già completato cliccando bullet o titolo step
205
+ * quando `progressStyle` è `'steps'`. @default true
206
+ */
207
+ allowStepIndicatorNavigation?: boolean;
196
208
  /** Label personalizzate per i pulsanti del wizard. */
197
209
  buttonLabels?: {
198
210
  previous?: string;
211
+ /** Label pulsante avanti; se assente si usa `submit`. */
199
212
  next?: string;
213
+ /**
214
+ * Alias di convenienza (come `submit` sul form builder) per il pulsante «avanti» / salva e prosegui.
215
+ * Usato solo se `next` non è valorizzato.
216
+ */
217
+ submit?: string;
200
218
  finish?: string;
201
219
  save?: string;
202
220
  };
@@ -1,4 +1,4 @@
1
- import { ChangeDetectionStrategy, Component, DestroyRef, Input, inject, ViewEncapsulation, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, DestroyRef, Input, inject, signal, ViewEncapsulation, } from '@angular/core';
2
2
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
3
  import { ActivatedRoute, NavigationEnd, Router } from '@angular/router';
4
4
  import { debounceTime, filter } from 'rxjs/operators';
@@ -14,6 +14,14 @@ import * as i0 from "@angular/core";
14
14
  * Optionally renders an automatic "back" button in the footer, derived from
15
15
  * the breadcrumb trail (navigates to the parent breadcrumb).
16
16
  *
17
+ * ### Route `data`
18
+ *
19
+ * Sulla rotta attiva (foglia) puoi impostare:
20
+ *
21
+ * - `breadcrumbOff: true` — nasconde `ui-page-header`
22
+ * - `footerNavOff: true` — nasconde `ui-base-layout__footer`
23
+ * - `id: string` — valorizza `data-route-id` sul contenitore `.ui-base-layout`
24
+ *
17
25
  * @selector ui-base-layout
18
26
  *
19
27
  * @example
@@ -65,6 +73,19 @@ export class UiBaseLayoutComponent {
65
73
  this.activatedRoute = inject(ActivatedRoute);
66
74
  this.breadcrumbService = inject(UiBreadcrumbService);
67
75
  this.destroyRef = inject(DestroyRef);
76
+ /**
77
+ * Flag derivati dalla catena attiva (`breadcrumbOff`, `footerNavOff`, `id`).
78
+ * Aggiornati a ogni `NavigationEnd`.
79
+ */
80
+ this.routeSnapshotFlags = signal({
81
+ breadcrumbOff: false,
82
+ footerNavOff: false,
83
+ routeId: null,
84
+ });
85
+ /** Header visibile se abilitato da input e non disattivato da `data.breadcrumbOff`. */
86
+ this.effectiveShowHeader = () => this.showHeader && !this.routeSnapshotFlags().breadcrumbOff;
87
+ /** Footer azioni visibile se ci sono azioni visibili e non disattivato da `data.footerNavOff`. */
88
+ this.effectiveShowFooterNav = () => !this.routeSnapshotFlags().footerNavOff && this.hasVisibleActions;
68
89
  }
69
90
  /**
70
91
  * Merged action list (back button + consumer actions).
@@ -92,25 +113,59 @@ export class UiBaseLayoutComponent {
92
113
  return this.mergedActions.some((a) => !a.hidden);
93
114
  }
94
115
  ngOnInit() {
95
- this.refreshBreadcrumbs();
116
+ this.syncFromActivatedRoute();
96
117
  this.router.events
97
118
  .pipe(filter((event) => event instanceof NavigationEnd), debounceTime(50), takeUntilDestroyed(this.destroyRef))
98
- .subscribe(() => this.refreshBreadcrumbs());
119
+ .subscribe(() => this.syncFromActivatedRoute());
99
120
  }
100
- /** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */
101
- refreshBreadcrumbs() {
121
+ /** @internal Allinea breadcrumb service e flag layout alla rotta corrente. */
122
+ syncFromActivatedRoute() {
123
+ this.refreshRouteSnapshotFlags();
102
124
  this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
103
125
  }
126
+ /** @internal Legge `breadcrumbOff`, `footerNavOff`, `id` dalla catena `pathFromRoot` / foglia. */
127
+ refreshRouteSnapshotFlags() {
128
+ const deepest = this.deepestChildRoute(this.activatedRoute);
129
+ let breadcrumbOff = false;
130
+ let footerNavOff = false;
131
+ for (const seg of deepest.pathFromRoot) {
132
+ const d = seg.snapshot.data;
133
+ if (!d)
134
+ continue;
135
+ // biome-ignore lint/complexity/useLiteralKeys: chiavi documentate in UiBaseLayoutRouteData
136
+ if (d['breadcrumbOff'] === true)
137
+ breadcrumbOff = true;
138
+ // biome-ignore lint/complexity/useLiteralKeys: chiavi documentate in UiBaseLayoutRouteData
139
+ if (d['footerNavOff'] === true)
140
+ footerNavOff = true;
141
+ }
142
+ let routeId = null;
143
+ let cursor = deepest;
144
+ while (cursor) {
145
+ const raw = cursor.snapshot.data?.['id'];
146
+ if (typeof raw === 'string' && raw.length > 0) {
147
+ routeId = raw;
148
+ break;
149
+ }
150
+ cursor = cursor.parent;
151
+ }
152
+ this.routeSnapshotFlags.set({ breadcrumbOff, footerNavOff, routeId });
153
+ }
154
+ /** @internal Percorre fino al `ActivatedRoute` foglia sotto il punto di iniezione. */
155
+ deepestChildRoute(route) {
156
+ let deepest = route;
157
+ while (deepest.firstChild) {
158
+ deepest = deepest.firstChild;
159
+ }
160
+ return deepest;
161
+ }
104
162
  /** @internal Determina se il back button deve essere nascosto per la rotta corrente. */
105
163
  shouldHideBackButton() {
106
164
  const currentPath = this.router.url.split('?')[0].split('#')[0];
107
165
  if (currentPath === this.homeRoute) {
108
166
  return true;
109
167
  }
110
- let deepestRoute = this.activatedRoute;
111
- while (deepestRoute.firstChild) {
112
- deepestRoute = deepestRoute.firstChild;
113
- }
168
+ const deepestRoute = this.deepestChildRoute(this.activatedRoute);
114
169
  // biome-ignore lint/complexity/useLiteralKeys: <explanation>
115
170
  if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {
116
171
  return true;
@@ -123,8 +178,8 @@ export class UiBaseLayoutComponent {
123
178
  }
124
179
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
125
180
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiBaseLayoutComponent, isStandalone: true, selector: "ui-base-layout", inputs: { showHeader: "showHeader", title: "title", homeRoute: "homeRoute", showHome: "showHome", updateDocumentTitle: "updateDocumentTitle", titleSuffix: "titleSuffix", showBackButton: "showBackButton", backButtonConfig: "backButtonConfig", actions: "actions", footerAlign: "footerAlign", footerGap: "footerGap" }, host: { classAttribute: "ui-base-layout-host" }, ngImport: i0, template: `
126
- <div class="ui-base-layout">
127
- @if (showHeader) {
181
+ <div class="ui-base-layout" [attr.data-route-id]="routeSnapshotFlags().routeId">
182
+ @if (effectiveShowHeader()) {
128
183
  <ui-page-header
129
184
  [title]="title"
130
185
  [homeRoute]="homeRoute"
@@ -138,7 +193,7 @@ export class UiBaseLayoutComponent {
138
193
  <ng-content />
139
194
  </main>
140
195
 
141
- @if (hasVisibleActions) {
196
+ @if (effectiveShowFooterNav()) {
142
197
  <aside
143
198
  class="ui-base-layout__footer"
144
199
  aria-label="Page actions"
@@ -160,8 +215,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
160
215
  args: [{ selector: 'ui-base-layout', standalone: true, imports: [UiPageHeaderComponent, UiButtonAreaComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
161
216
  class: 'ui-base-layout-host',
162
217
  }, template: `
163
- <div class="ui-base-layout">
164
- @if (showHeader) {
218
+ <div class="ui-base-layout" [attr.data-route-id]="routeSnapshotFlags().routeId">
219
+ @if (effectiveShowHeader()) {
165
220
  <ui-page-header
166
221
  [title]="title"
167
222
  [homeRoute]="homeRoute"
@@ -175,7 +230,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
175
230
  <ng-content />
176
231
  </main>
177
232
 
178
- @if (hasVisibleActions) {
233
+ @if (effectiveShowFooterNav()) {
179
234
  <aside
180
235
  class="ui-base-layout__footer"
181
236
  aria-label="Page actions"
@@ -214,4 +269,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
214
269
  }], footerGap: [{
215
270
  type: Input
216
271
  }] } });
217
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-layout.component.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/base-layout/base-layout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  DestroyRef,\r\n  Input,\r\n  inject,\r\n  type OnInit,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\n\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { debounceTime, filter } from 'rxjs/operators';\r\nimport type { UiButtonAreaAlign, UiButtonDescriptor } from '../button/button.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport type { UiBackButtonConfig } from './base-layout.types';\r\n\r\n/**\r\n * Full-page layout component that combines a page header (breadcrumbs + title),\r\n * a main content area via `<ng-content>`, and a sticky footer action bar\r\n * powered by `UiButtonAreaComponent`.\r\n *\r\n * Optionally renders an automatic \"back\" button in the footer, derived from\r\n * the breadcrumb trail (navigates to the parent breadcrumb).\r\n *\r\n * @selector ui-base-layout\r\n *\r\n * @example\r\n * ```html\r\n * <ui-base-layout [actions]=\"pageActions\" footerAlign=\"end\">\r\n *   <p>Your page content here</p>\r\n * </ui-base-layout>\r\n * ```\r\n *\r\n * @example\r\n * ```html\r\n * <!-- With back button and custom title -->\r\n * <ui-base-layout\r\n *   title=\"Dettaglio utente\"\r\n *   [showBackButton]=\"true\"\r\n *   [actions]=\"[\r\n *     { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },\r\n *   ]\"\r\n * >\r\n *   <app-user-detail />\r\n * </ui-base-layout>\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-base-layout',\r\n  standalone: true,\r\n  imports: [UiPageHeaderComponent, UiButtonAreaComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-base-layout-host',\r\n  },\r\n  template: `\r\n    <div class=\"ui-base-layout\">\r\n      @if (showHeader) {\r\n        <ui-page-header\r\n          [title]=\"title\"\r\n          [homeRoute]=\"homeRoute\"\r\n          [showHome]=\"showHome\"\r\n          [updateDocumentTitle]=\"updateDocumentTitle\"\r\n          [titleSuffix]=\"titleSuffix\"\r\n        />\r\n      }\r\n\r\n      <main class=\"ui-base-layout__content\">\r\n        <ng-content />\r\n      </main>\r\n\r\n      @if (hasVisibleActions) {\r\n        <aside\r\n          class=\"ui-base-layout__footer\"\r\n          aria-label=\"Page actions\"\r\n        >\r\n          <ui-button-area\r\n            [buttons]=\"mergedActions\"\r\n            [align]=\"footerAlign\"\r\n            [gap]=\"footerGap\"\r\n            [stackOnMobile]=\"true\"\r\n            ariaLabel=\"Page actions\"\r\n          />\r\n        </aside>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './base-layout.component.scss',\r\n})\r\nexport class UiBaseLayoutComponent implements OnInit {\r\n  /** Whether to display the page header (breadcrumbs + title). */\r\n  @Input() showHeader = true;\r\n\r\n  /** Override the auto-detected page title. Passed through to `ui-page-header`. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link in the breadcrumb trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** Whether to show an automatic \"back\" button based on breadcrumbs. */\r\n  @Input() showBackButton = true;\r\n\r\n  /** Configuration overrides for the automatic back button. */\r\n  @Input() backButtonConfig: UiBackButtonConfig = {};\r\n\r\n  /** Array of button descriptors for the footer action area. */\r\n  @Input() actions: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the footer button area. */\r\n  @Input() footerAlign: UiButtonAreaAlign = 'start';\r\n\r\n  /** Gap size between footer buttons. */\r\n  @Input() footerGap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'sm';\r\n\r\n  private readonly router = inject(Router);\r\n\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n\r\n\r\n\r\n  /**\r\n   * Merged action list (back button + consumer actions).\r\n   * Computed on every change-detection check so it stays in sync\r\n   * with both navigation state (breadcrumbs) and input changes.\r\n   */\r\n  get mergedActions(): UiButtonDescriptor[] {\r\n    const result: UiButtonDescriptor[] = [];\r\n\r\n    if (this.showBackButton) {\r\n      result.push({\r\n        id: '__ui-back',\r\n        label: this.backButtonConfig.label ?? 'Indietro',\r\n        icon: (this.backButtonConfig.icon as any) ?? 'arrow-left',\r\n        iconPosition: 'leading',\r\n        variant: (this.backButtonConfig.variant as any) ?? 'outline',\r\n        hidden: this.shouldHideBackButton(),\r\n        action: () => this.navigateBack(),\r\n      });\r\n    }\r\n\r\n    result.push(...this.actions);\r\n    return result;\r\n  }\r\n\r\n  /** Whether the footer should render (at least one non-hidden action exists). */\r\n  get hasVisibleActions(): boolean {\r\n    return this.mergedActions.some((a) => !a.hidden);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.refreshBreadcrumbs();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        debounceTime(50),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.refreshBreadcrumbs());\r\n  }\r\n\r\n  /** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */\r\n  private refreshBreadcrumbs(): void {\r\n    this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Determina se il back button deve essere nascosto per la rotta corrente. */\r\n  private shouldHideBackButton(): boolean {\r\n    const currentPath = this.router.url.split('?')[0].split('#')[0];\r\n    if (currentPath === this.homeRoute) {\r\n      return true;\r\n    }\r\n\r\n    let deepestRoute = this.activatedRoute;\r\n    while (deepestRoute.firstChild) {\r\n      deepestRoute = deepestRoute.firstChild;\r\n    }\r\n\r\n    // biome-ignore lint/complexity/useLiteralKeys: <explanation>\r\n    if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {\r\n      return true;\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  /** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */\r\n  private navigateBack(): void {\r\n    this.breadcrumbService.navigateBack(this.homeRoute);\r\n  }\r\n}\r\n"]}
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-layout.component.js","sourceRoot":"","sources":["../../../../../../../packages/ng-ui-system/src/lib/components/base-layout/base-layout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  DestroyRef,\r\n  Input,\r\n  inject,\r\n  signal,\r\n  type OnInit,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\n\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { debounceTime, filter } from 'rxjs/operators';\r\nimport type { UiButtonAreaAlign, UiButtonDescriptor } from '../button/button.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport type { UiBackButtonConfig } from './base-layout.types';\r\n\r\n/** Stato letto dalla rotta foglia per header/footer opzionali e `data-route-id`. */\r\ninterface UiBaseLayoutRouteSnapshotFlags {\r\n  breadcrumbOff: boolean;\r\n  footerNavOff: boolean;\r\n  routeId: string | null;\r\n}\r\n\r\n/**\r\n * Full-page layout component that combines a page header (breadcrumbs + title),\r\n * a main content area via `<ng-content>`, and a sticky footer action bar\r\n * powered by `UiButtonAreaComponent`.\r\n *\r\n * Optionally renders an automatic \"back\" button in the footer, derived from\r\n * the breadcrumb trail (navigates to the parent breadcrumb).\r\n *\r\n * ### Route `data`\r\n *\r\n * Sulla rotta attiva (foglia) puoi impostare:\r\n *\r\n * - `breadcrumbOff: true` — nasconde `ui-page-header`\r\n * - `footerNavOff: true` — nasconde `ui-base-layout__footer`\r\n * - `id: string` — valorizza `data-route-id` sul contenitore `.ui-base-layout`\r\n *\r\n * @selector ui-base-layout\r\n *\r\n * @example\r\n * ```html\r\n * <ui-base-layout [actions]=\"pageActions\" footerAlign=\"end\">\r\n *   <p>Your page content here</p>\r\n * </ui-base-layout>\r\n * ```\r\n *\r\n * @example\r\n * ```html\r\n * <!-- With back button and custom title -->\r\n * <ui-base-layout\r\n *   title=\"Dettaglio utente\"\r\n *   [showBackButton]=\"true\"\r\n *   [actions]=\"[\r\n *     { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },\r\n *   ]\"\r\n * >\r\n *   <app-user-detail />\r\n * </ui-base-layout>\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-base-layout',\r\n  standalone: true,\r\n  imports: [UiPageHeaderComponent, UiButtonAreaComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-base-layout-host',\r\n  },\r\n  template: `\r\n    <div class=\"ui-base-layout\" [attr.data-route-id]=\"routeSnapshotFlags().routeId\">\r\n      @if (effectiveShowHeader()) {\r\n        <ui-page-header\r\n          [title]=\"title\"\r\n          [homeRoute]=\"homeRoute\"\r\n          [showHome]=\"showHome\"\r\n          [updateDocumentTitle]=\"updateDocumentTitle\"\r\n          [titleSuffix]=\"titleSuffix\"\r\n        />\r\n      }\r\n\r\n      <main class=\"ui-base-layout__content\">\r\n        <ng-content />\r\n      </main>\r\n\r\n      @if (effectiveShowFooterNav()) {\r\n        <aside\r\n          class=\"ui-base-layout__footer\"\r\n          aria-label=\"Page actions\"\r\n        >\r\n          <ui-button-area\r\n            [buttons]=\"mergedActions\"\r\n            [align]=\"footerAlign\"\r\n            [gap]=\"footerGap\"\r\n            [stackOnMobile]=\"true\"\r\n            ariaLabel=\"Page actions\"\r\n          />\r\n        </aside>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './base-layout.component.scss',\r\n})\r\nexport class UiBaseLayoutComponent implements OnInit {\r\n  /** Whether to display the page header (breadcrumbs + title). */\r\n  @Input() showHeader = true;\r\n\r\n  /** Override the auto-detected page title. Passed through to `ui-page-header`. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link in the breadcrumb trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** Whether to show an automatic \"back\" button based on breadcrumbs. */\r\n  @Input() showBackButton = true;\r\n\r\n  /** Configuration overrides for the automatic back button. */\r\n  @Input() backButtonConfig: UiBackButtonConfig = {};\r\n\r\n  /** Array of button descriptors for the footer action area. */\r\n  @Input() actions: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the footer button area. */\r\n  @Input() footerAlign: UiButtonAreaAlign = 'start';\r\n\r\n  /** Gap size between footer buttons. */\r\n  @Input() footerGap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'sm';\r\n\r\n  private readonly router = inject(Router);\r\n\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n\r\n  /**\r\n   * Flag derivati dalla catena attiva (`breadcrumbOff`, `footerNavOff`, `id`).\r\n   * Aggiornati a ogni `NavigationEnd`.\r\n   */\r\n  protected readonly routeSnapshotFlags = signal<UiBaseLayoutRouteSnapshotFlags>({\r\n    breadcrumbOff: false,\r\n    footerNavOff: false,\r\n    routeId: null,\r\n  });\r\n\r\n  /** Header visibile se abilitato da input e non disattivato da `data.breadcrumbOff`. */\r\n  protected readonly effectiveShowHeader = () =>\r\n    this.showHeader && !this.routeSnapshotFlags().breadcrumbOff;\r\n\r\n  /** Footer azioni visibile se ci sono azioni visibili e non disattivato da `data.footerNavOff`. */\r\n  protected readonly effectiveShowFooterNav = () =>\r\n    !this.routeSnapshotFlags().footerNavOff && this.hasVisibleActions;\r\n\r\n  /**\r\n   * Merged action list (back button + consumer actions).\r\n   * Computed on every change-detection check so it stays in sync\r\n   * with both navigation state (breadcrumbs) and input changes.\r\n   */\r\n  get mergedActions(): UiButtonDescriptor[] {\r\n    const result: UiButtonDescriptor[] = [];\r\n\r\n    if (this.showBackButton) {\r\n      result.push({\r\n        id: '__ui-back',\r\n        label: this.backButtonConfig.label ?? 'Indietro',\r\n        icon: (this.backButtonConfig.icon as any) ?? 'arrow-left',\r\n        iconPosition: 'leading',\r\n        variant: (this.backButtonConfig.variant as any) ?? 'outline',\r\n        hidden: this.shouldHideBackButton(),\r\n        action: () => this.navigateBack(),\r\n      });\r\n    }\r\n\r\n    result.push(...this.actions);\r\n    return result;\r\n  }\r\n\r\n  /** Whether the footer should render (at least one non-hidden action exists). */\r\n  get hasVisibleActions(): boolean {\r\n    return this.mergedActions.some((a) => !a.hidden);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.syncFromActivatedRoute();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        debounceTime(50),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.syncFromActivatedRoute());\r\n  }\r\n\r\n  /** @internal Allinea breadcrumb service e flag layout alla rotta corrente. */\r\n  private syncFromActivatedRoute(): void {\r\n    this.refreshRouteSnapshotFlags();\r\n    this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Legge `breadcrumbOff`, `footerNavOff`, `id` dalla catena `pathFromRoot` / foglia. */\r\n  private refreshRouteSnapshotFlags(): void {\r\n    const deepest = this.deepestChildRoute(this.activatedRoute);\r\n\r\n    let breadcrumbOff = false;\r\n    let footerNavOff = false;\r\n    for (const seg of deepest.pathFromRoot) {\r\n      const d = seg.snapshot.data as Record<string, unknown> | undefined;\r\n      if (!d) continue;\r\n      // biome-ignore lint/complexity/useLiteralKeys: chiavi documentate in UiBaseLayoutRouteData\r\n      if (d['breadcrumbOff'] === true) breadcrumbOff = true;\r\n      // biome-ignore lint/complexity/useLiteralKeys: chiavi documentate in UiBaseLayoutRouteData\r\n      if (d['footerNavOff'] === true) footerNavOff = true;\r\n    }\r\n\r\n    let routeId: string | null = null;\r\n    let cursor: ActivatedRoute | null = deepest;\r\n    while (cursor) {\r\n      const raw = cursor.snapshot.data?.['id'];\r\n      if (typeof raw === 'string' && raw.length > 0) {\r\n        routeId = raw;\r\n        break;\r\n      }\r\n      cursor = cursor.parent;\r\n    }\r\n\r\n    this.routeSnapshotFlags.set({ breadcrumbOff, footerNavOff, routeId });\r\n  }\r\n\r\n  /** @internal Percorre fino al `ActivatedRoute` foglia sotto il punto di iniezione. */\r\n  private deepestChildRoute(route: ActivatedRoute): ActivatedRoute {\r\n    let deepest = route;\r\n    while (deepest.firstChild) {\r\n      deepest = deepest.firstChild;\r\n    }\r\n    return deepest;\r\n  }\r\n\r\n  /** @internal Determina se il back button deve essere nascosto per la rotta corrente. */\r\n  private shouldHideBackButton(): boolean {\r\n    const currentPath = this.router.url.split('?')[0].split('#')[0];\r\n    if (currentPath === this.homeRoute) {\r\n      return true;\r\n    }\r\n\r\n    const deepestRoute = this.deepestChildRoute(this.activatedRoute);\r\n\r\n    // biome-ignore lint/complexity/useLiteralKeys: <explanation>\r\n    if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {\r\n      return true;\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  /** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */\r\n  private navigateBack(): void {\r\n    this.breadcrumbService.navigateBack(this.homeRoute);\r\n  }\r\n}\r\n"]}
@@ -3,4 +3,4 @@
3
3
  * Types and interfaces for UiBaseLayout component.
4
4
  */
5
5
  export {};
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1sYXlvdXQudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2Jhc2UtbGF5b3V0LnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAbW9kdWxlIG5nLXVpLXN5c3RlbS9iYXNlLWxheW91dFxyXG4gKiBUeXBlcyBhbmQgaW50ZXJmYWNlcyBmb3IgVWlCYXNlTGF5b3V0IGNvbXBvbmVudC5cclxuICovXHJcblxyXG5pbXBvcnQgeyBVaUJ1dHRvbkRlc2NyaXB0b3IsIFVpQnV0dG9uQXJlYUFsaWduIH0gZnJvbSAnLi4vYnV0dG9uL2J1dHRvbi50eXBlcyc7XHJcblxyXG4vLyBSZS1leHBvcnQgZm9yIGNvbnN1bWVyIGNvbnZlbmllbmNlXHJcbmV4cG9ydCB7IFVpQnV0dG9uRGVzY3JpcHRvciwgVWlCdXR0b25BcmVhQWxpZ24gfTtcclxuXHJcbi8qKlxyXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgYXV0b21hdGljIFwiYmFja1wiIGJ1dHRvbiByZW5kZXJlZCBpbiB0aGUgZm9vdGVyIGFjdGlvbiBhcmVhLlxyXG4gKlxyXG4gKiBAdXNhZ2VOb3Rlc1xyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGNvbnN0IGJhY2tDb25maWc6IFVpQmFja0J1dHRvbkNvbmZpZyA9IHtcclxuICogICBsYWJlbDogJ0luZGlldHJvJyxcclxuICogICBpY29uOiAnYXJyb3ctbGVmdCcsXHJcbiAqICAgdmFyaWFudDogJ291dGxpbmUnLFxyXG4gKiB9O1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWlCYWNrQnV0dG9uQ29uZmlnIHtcclxuICAvKiogTGFiZWwgdGV4dCBmb3IgdGhlIGJhY2sgYnV0dG9uLiBAZGVmYXVsdCAnSW5kaWV0cm8nICovXHJcbiAgbGFiZWw/OiBzdHJpbmc7XHJcbiAgLyoqIEx1Y2lkZSBpY29uIG5hbWUuIEBkZWZhdWx0ICdhcnJvdy1sZWZ0JyAqL1xyXG4gIGljb24/OiBzdHJpbmc7XHJcbiAgLyoqIFZpc3VhbCB2YXJpYW50LiBAZGVmYXVsdCAnb3V0bGluZScgKi9cclxuICB2YXJpYW50Pzogc3RyaW5nO1xyXG59XHJcbiJdfQ==
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1sYXlvdXQudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2Jhc2UtbGF5b3V0LnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAbW9kdWxlIG5nLXVpLXN5c3RlbS9iYXNlLWxheW91dFxyXG4gKiBUeXBlcyBhbmQgaW50ZXJmYWNlcyBmb3IgVWlCYXNlTGF5b3V0IGNvbXBvbmVudC5cclxuICovXHJcblxyXG5pbXBvcnQgeyBVaUJ1dHRvbkRlc2NyaXB0b3IsIFVpQnV0dG9uQXJlYUFsaWduIH0gZnJvbSAnLi4vYnV0dG9uL2J1dHRvbi50eXBlcyc7XHJcblxyXG4vLyBSZS1leHBvcnQgZm9yIGNvbnN1bWVyIGNvbnZlbmllbmNlXHJcbmV4cG9ydCB7IFVpQnV0dG9uRGVzY3JpcHRvciwgVWlCdXR0b25BcmVhQWxpZ24gfTtcclxuXHJcbi8qKlxyXG4gKiBDaGlhdmkgb3B6aW9uYWxpIGRpIGBkYXRhYCBzdWxsYSByb3R0YSBhdHRpdmEgcmljb25vc2NpdXRlIGRhIGBVaUJhc2VMYXlvdXRDb21wb25lbnRgLlxyXG4gKlxyXG4gKiAtIGBpZGA6IHN0ZXNzYSByaXNhbGl0YSB1c2F0YSBkYWkgYnJlYWRjcnVtYiAoZGFsIHNlZ21lbnRvIGZvZ2xpYSB2ZXJzbyBpbCByb290OiBwcmltbyBgaWRgIHRyb3ZhdG8pLlxyXG4gKiAtIGBicmVhZGNydW1iT2ZmYCAvIGBmb290ZXJOYXZPZmZgOiBzZSBgdHJ1ZWAgc3UgKipxdWFsc2lhc2kqKiBzZWdtZW50byBpbiBgcGF0aEZyb21Sb290YCwgbOKAmWFyZWEgdmllbmUgbmFzY29zdGEuXHJcbiAqL1xyXG5leHBvcnQgaW50ZXJmYWNlIFVpQmFzZUxheW91dFJvdXRlRGF0YSB7XHJcbiAgLyoqIElkZW50aWZpY2F0aXZvIHN0YWJpbGUgZGVsbGEgcm90dGE7IHJlc28gY29tZSBgZGF0YS1yb3V0ZS1pZGAgc3VsIHdyYXBwZXIgYC51aS1iYXNlLWxheW91dGAuICovXHJcbiAgaWQ/OiBzdHJpbmc7XHJcbiAgLyoqIFNlIGB0cnVlYCwgbm9uIHZpZW5lIHJlbmRlcml6emF0byBgdWktcGFnZS1oZWFkZXJgIChicmVhZGNydW1iIGUgdGl0b2xvKS4gKi9cclxuICBicmVhZGNydW1iT2ZmPzogYm9vbGVhbjtcclxuICAvKiogU2UgYHRydWVgLCBub24gdmllbmUgcmVuZGVyaXp6YXRhIGxhIGJhcnJhIGB1aS1iYXNlLWxheW91dF9fZm9vdGVyYCAoYXppb25pIC8gaW5kaWV0cm8pLiAqL1xyXG4gIGZvb3Rlck5hdk9mZj86IGJvb2xlYW47XHJcbn1cclxuXHJcbi8qKlxyXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgYXV0b21hdGljIFwiYmFja1wiIGJ1dHRvbiByZW5kZXJlZCBpbiB0aGUgZm9vdGVyIGFjdGlvbiBhcmVhLlxyXG4gKlxyXG4gKiBAdXNhZ2VOb3Rlc1xyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGNvbnN0IGJhY2tDb25maWc6IFVpQmFja0J1dHRvbkNvbmZpZyA9IHtcclxuICogICBsYWJlbDogJ0luZGlldHJvJyxcclxuICogICBpY29uOiAnYXJyb3ctbGVmdCcsXHJcbiAqICAgdmFyaWFudDogJ291dGxpbmUnLFxyXG4gKiB9O1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWlCYWNrQnV0dG9uQ29uZmlnIHtcclxuICAvKiogTGFiZWwgdGV4dCBmb3IgdGhlIGJhY2sgYnV0dG9uLiBAZGVmYXVsdCAnSW5kaWV0cm8nICovXHJcbiAgbGFiZWw/OiBzdHJpbmc7XHJcbiAgLyoqIEx1Y2lkZSBpY29uIG5hbWUuIEBkZWZhdWx0ICdhcnJvdy1sZWZ0JyAqL1xyXG4gIGljb24/OiBzdHJpbmc7XHJcbiAgLyoqIFZpc3VhbCB2YXJpYW50LiBAZGVmYXVsdCAnb3V0bGluZScgKi9cclxuICB2YXJpYW50Pzogc3RyaW5nO1xyXG59XHJcbiJdfQ==
@@ -11,4 +11,4 @@
11
11
  */
12
12
  // Components
13
13
  export { UiBaseLayoutComponent } from './base-layout.component';
14
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCxhQUFhO0FBQ2IsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogbmctdWktc3lzdGVtIOKAlCBCYXNlIExheW91dCBlbnRyeSBwb2ludC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBpbXBvcnQge1xyXG4gKiAgIFVpQmFzZUxheW91dENvbXBvbmVudCxcclxuICogICBVaUJhY2tCdXR0b25Db25maWcsXHJcbiAqIH0gZnJvbSAnbmctdWktc3lzdGVtJztcclxuICogYGBgXHJcbiAqL1xyXG5cclxuLy8gQ29tcG9uZW50c1xyXG5leHBvcnQgeyBVaUJhc2VMYXlvdXRDb21wb25lbnQgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LmNvbXBvbmVudCc7XHJcblxyXG4vLyBUeXBlc1xyXG5leHBvcnQgeyBVaUJhY2tCdXR0b25Db25maWcgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LnR5cGVzJztcclxuIl19
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCxhQUFhO0FBQ2IsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogbmctdWktc3lzdGVtIOKAlCBCYXNlIExheW91dCBlbnRyeSBwb2ludC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBpbXBvcnQge1xyXG4gKiAgIFVpQmFzZUxheW91dENvbXBvbmVudCxcclxuICogICBVaUJhY2tCdXR0b25Db25maWcsXHJcbiAqIH0gZnJvbSAnbmctdWktc3lzdGVtJztcclxuICogYGBgXHJcbiAqL1xyXG5cclxuLy8gQ29tcG9uZW50c1xyXG5leHBvcnQgeyBVaUJhc2VMYXlvdXRDb21wb25lbnQgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LmNvbXBvbmVudCc7XHJcblxyXG4vLyBUeXBlc1xyXG5leHBvcnQgeyBVaUJhY2tCdXR0b25Db25maWcsIFVpQmFzZUxheW91dFJvdXRlRGF0YSB9IGZyb20gJy4vYmFzZS1sYXlvdXQudHlwZXMnO1xyXG4iXX0=