@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,218 @@
1
+ import { Component, DestroyRef, inject, Input, ChangeDetectionStrategy, ViewEncapsulation, } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { Location } from '@angular/common';
4
+ import { ActivatedRoute, NavigationEnd, Router } from '@angular/router';
5
+ import { filter, debounceTime } from 'rxjs/operators';
6
+ import { UiPageHeaderComponent } from '../page-header/page-header.component';
7
+ import { UiBreadcrumbService } from '../page-header/breadcrumb.service';
8
+ import { UiButtonAreaComponent } from '../button/button-area.component';
9
+ import * as i0 from "@angular/core";
10
+ /**
11
+ * Full-page layout component that combines a page header (breadcrumbs + title),
12
+ * a main content area via `<ng-content>`, and a sticky footer action bar
13
+ * powered by `UiButtonAreaComponent`.
14
+ *
15
+ * Optionally renders an automatic "back" button in the footer, derived from
16
+ * the breadcrumb trail (navigates to the parent breadcrumb).
17
+ *
18
+ * @selector ui-base-layout
19
+ *
20
+ * @example
21
+ * ```html
22
+ * <ui-base-layout [actions]="pageActions" footerAlign="end">
23
+ * <p>Your page content here</p>
24
+ * </ui-base-layout>
25
+ * ```
26
+ *
27
+ * @example
28
+ * ```html
29
+ * <!-- With back button and custom title -->
30
+ * <ui-base-layout
31
+ * title="Dettaglio utente"
32
+ * [showBackButton]="true"
33
+ * [actions]="[
34
+ * { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },
35
+ * ]"
36
+ * >
37
+ * <app-user-detail />
38
+ * </ui-base-layout>
39
+ * ```
40
+ */
41
+ export class UiBaseLayoutComponent {
42
+ constructor() {
43
+ /** Whether to display the page header (breadcrumbs + title). */
44
+ this.showHeader = true;
45
+ /** Override the auto-detected page title. Passed through to `ui-page-header`. */
46
+ this.title = '';
47
+ /** Route path for the Home breadcrumb link. */
48
+ this.homeRoute = '/';
49
+ /** Whether to display the Home link in the breadcrumb trail. */
50
+ this.showHome = true;
51
+ /** When `true`, updates `document.title` on navigation. */
52
+ this.updateDocumentTitle = false;
53
+ /** Suffix appended to `document.title`. */
54
+ this.titleSuffix = '';
55
+ /** Whether to show an automatic "back" button based on breadcrumbs. */
56
+ this.showBackButton = true;
57
+ /** Configuration overrides for the automatic back button. */
58
+ this.backButtonConfig = {};
59
+ /** Array of button descriptors for the footer action area. */
60
+ this.actions = [];
61
+ /** Horizontal alignment of the footer button area. */
62
+ this.footerAlign = 'start';
63
+ /** Gap size between footer buttons. */
64
+ this.footerGap = 'sm';
65
+ this.router = inject(Router);
66
+ this.location = inject(Location);
67
+ this.activatedRoute = inject(ActivatedRoute);
68
+ this.breadcrumbService = inject(UiBreadcrumbService);
69
+ this.destroyRef = inject(DestroyRef);
70
+ /** @internal Current breadcrumbs, updated on each navigation. */
71
+ this.breadcrumbs = [];
72
+ }
73
+ /**
74
+ * Merged action list (back button + consumer actions).
75
+ * Computed on every change-detection check so it stays in sync
76
+ * with both navigation state (breadcrumbs) and input changes.
77
+ */
78
+ get mergedActions() {
79
+ const result = [];
80
+ if (this.showBackButton) {
81
+ const backTarget = this.getBackTarget();
82
+ result.push({
83
+ id: '__ui-back',
84
+ label: this.backButtonConfig.label ?? 'Indietro',
85
+ icon: this.backButtonConfig.icon ?? 'arrow-left',
86
+ iconPosition: 'leading',
87
+ variant: this.backButtonConfig.variant ?? 'outline',
88
+ hidden: !backTarget,
89
+ action: () => this.navigateBack(),
90
+ });
91
+ }
92
+ result.push(...this.actions);
93
+ return result;
94
+ }
95
+ /** Whether the footer should render (at least one non-hidden action exists). */
96
+ get hasVisibleActions() {
97
+ return this.mergedActions.some((a) => !a.hidden);
98
+ }
99
+ ngOnInit() {
100
+ this.refreshBreadcrumbs();
101
+ this.router.events
102
+ .pipe(filter((event) => event instanceof NavigationEnd), debounceTime(50), takeUntilDestroyed(this.destroyRef))
103
+ .subscribe(() => this.refreshBreadcrumbs());
104
+ }
105
+ /** @internal Refreshes breadcrumb state from the current route. */
106
+ refreshBreadcrumbs() {
107
+ this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
108
+ }
109
+ /** @internal Determines the back navigation target from breadcrumbs. */
110
+ getBackTarget() {
111
+ if (this.breadcrumbs.length < 2)
112
+ return null;
113
+ return this.breadcrumbs[this.breadcrumbs.length - 2];
114
+ }
115
+ /** @internal Navigates to the parent breadcrumb or falls back to browser history. */
116
+ navigateBack() {
117
+ const target = this.getBackTarget();
118
+ if (target?.url) {
119
+ this.router.navigateByUrl(target.url);
120
+ }
121
+ else {
122
+ this.location.back();
123
+ }
124
+ }
125
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
126
+ 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: `
127
+ <div class="ui-base-layout">
128
+ @if (showHeader) {
129
+ <ui-page-header
130
+ [title]="title"
131
+ [homeRoute]="homeRoute"
132
+ [showHome]="showHome"
133
+ [updateDocumentTitle]="updateDocumentTitle"
134
+ [titleSuffix]="titleSuffix"
135
+ />
136
+ }
137
+
138
+ <main class="ui-base-layout__content">
139
+ <ng-content />
140
+ </main>
141
+
142
+ @if (hasVisibleActions) {
143
+ <aside
144
+ class="ui-base-layout__footer"
145
+ aria-label="Page actions"
146
+ >
147
+ <ui-button-area
148
+ [buttons]="mergedActions"
149
+ [align]="footerAlign"
150
+ [gap]="footerGap"
151
+ [stackOnMobile]="true"
152
+ ariaLabel="Page actions"
153
+ />
154
+ </aside>
155
+ }
156
+ </div>
157
+ `, isInline: true, styles: [".ui-base-layout-host{display:block;min-height:100%}.ui-base-layout{display:flex;flex-direction:column;min-height:100%;font-family:var(--ui-font-family);padding:var(--ui-spacing-6)}.ui-base-layout__content{flex:1 1 auto}.ui-base-layout__footer{position:sticky;bottom:0;z-index:var(--ui-z-sticky, 100);padding:var(--ui-spacing-4) 0;margin-top:var(--ui-spacing-6);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border)}@media (min-width: 640px){.ui-base-layout__footer{padding:var(--ui-spacing-3) 0}}\n"], dependencies: [{ kind: "component", type: UiPageHeaderComponent, selector: "ui-page-header", inputs: ["title", "homeRoute", "showHome", "updateDocumentTitle", "titleSuffix"] }, { kind: "component", type: UiButtonAreaComponent, selector: "ui-button-area", inputs: ["buttons", "align", "ariaLabel", "gap", "stackOnMobile", "disableWhileLoading", "loadingIds"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
158
+ }
159
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, decorators: [{
160
+ type: Component,
161
+ args: [{ selector: 'ui-base-layout', standalone: true, imports: [UiPageHeaderComponent, UiButtonAreaComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
162
+ class: 'ui-base-layout-host',
163
+ }, template: `
164
+ <div class="ui-base-layout">
165
+ @if (showHeader) {
166
+ <ui-page-header
167
+ [title]="title"
168
+ [homeRoute]="homeRoute"
169
+ [showHome]="showHome"
170
+ [updateDocumentTitle]="updateDocumentTitle"
171
+ [titleSuffix]="titleSuffix"
172
+ />
173
+ }
174
+
175
+ <main class="ui-base-layout__content">
176
+ <ng-content />
177
+ </main>
178
+
179
+ @if (hasVisibleActions) {
180
+ <aside
181
+ class="ui-base-layout__footer"
182
+ aria-label="Page actions"
183
+ >
184
+ <ui-button-area
185
+ [buttons]="mergedActions"
186
+ [align]="footerAlign"
187
+ [gap]="footerGap"
188
+ [stackOnMobile]="true"
189
+ ariaLabel="Page actions"
190
+ />
191
+ </aside>
192
+ }
193
+ </div>
194
+ `, styles: [".ui-base-layout-host{display:block;min-height:100%}.ui-base-layout{display:flex;flex-direction:column;min-height:100%;font-family:var(--ui-font-family);padding:var(--ui-spacing-6)}.ui-base-layout__content{flex:1 1 auto}.ui-base-layout__footer{position:sticky;bottom:0;z-index:var(--ui-z-sticky, 100);padding:var(--ui-spacing-4) 0;margin-top:var(--ui-spacing-6);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border)}@media (min-width: 640px){.ui-base-layout__footer{padding:var(--ui-spacing-3) 0}}\n"] }]
195
+ }], propDecorators: { showHeader: [{
196
+ type: Input
197
+ }], title: [{
198
+ type: Input
199
+ }], homeRoute: [{
200
+ type: Input
201
+ }], showHome: [{
202
+ type: Input
203
+ }], updateDocumentTitle: [{
204
+ type: Input
205
+ }], titleSuffix: [{
206
+ type: Input
207
+ }], showBackButton: [{
208
+ type: Input
209
+ }], backButtonConfig: [{
210
+ type: Input
211
+ }], actions: [{
212
+ type: Input
213
+ }], footerAlign: [{
214
+ type: Input
215
+ }], footerGap: [{
216
+ type: Input
217
+ }] } });
218
+ //# 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  Component,\r\n  DestroyRef,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  ChangeDetectionStrategy,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { Location } from '@angular/common';\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { filter, debounceTime } from 'rxjs/operators';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiBreadcrumbItem } from '../page-header/page-header.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiButtonDescriptor, UiButtonAreaAlign } from '../button/button.types';\r\nimport { 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  private readonly location = inject(Location);\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  /** @internal Current breadcrumbs, updated on each navigation. */\r\n  private breadcrumbs: UiBreadcrumbItem[] = [];\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      const backTarget = this.getBackTarget();\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: !backTarget,\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 Refreshes breadcrumb state from the current route. */\r\n  private refreshBreadcrumbs(): void {\r\n    this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Determines the back navigation target from breadcrumbs. */\r\n  private getBackTarget(): UiBreadcrumbItem | null {\r\n    if (this.breadcrumbs.length < 2) return null;\r\n    return this.breadcrumbs[this.breadcrumbs.length - 2];\r\n  }\r\n\r\n  /** @internal Navigates to the parent breadcrumb or falls back to browser history. */\r\n  private navigateBack(): void {\r\n    const target = this.getBackTarget();\r\n    if (target?.url) {\r\n      this.router.navigateByUrl(target.url);\r\n    } else {\r\n      this.location.back();\r\n    }\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,6 @@
1
+ /**
2
+ * @module ng-ui-system/base-layout
3
+ * Types and interfaces for UiBaseLayout component.
4
+ */
5
+ export {};
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1sYXlvdXQudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2Jhc2UtbGF5b3V0LnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAbW9kdWxlIG5nLXVpLXN5c3RlbS9iYXNlLWxheW91dFxyXG4gKiBUeXBlcyBhbmQgaW50ZXJmYWNlcyBmb3IgVWlCYXNlTGF5b3V0IGNvbXBvbmVudC5cclxuICovXHJcblxyXG5pbXBvcnQgeyBVaUJ1dHRvbkRlc2NyaXB0b3IsIFVpQnV0dG9uQXJlYUFsaWduIH0gZnJvbSAnLi4vYnV0dG9uL2J1dHRvbi50eXBlcyc7XHJcblxyXG4vLyBSZS1leHBvcnQgZm9yIGNvbnN1bWVyIGNvbnZlbmllbmNlXHJcbmV4cG9ydCB7IFVpQnV0dG9uRGVzY3JpcHRvciwgVWlCdXR0b25BcmVhQWxpZ24gfTtcclxuXHJcbi8qKlxyXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgYXV0b21hdGljIFwiYmFja1wiIGJ1dHRvbiByZW5kZXJlZCBpbiB0aGUgZm9vdGVyIGFjdGlvbiBhcmVhLlxyXG4gKlxyXG4gKiBAdXNhZ2VOb3Rlc1xyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGNvbnN0IGJhY2tDb25maWc6IFVpQmFja0J1dHRvbkNvbmZpZyA9IHtcclxuICogICBsYWJlbDogJ0luZGlldHJvJyxcclxuICogICBpY29uOiAnYXJyb3ctbGVmdCcsXHJcbiAqICAgdmFyaWFudDogJ291dGxpbmUnLFxyXG4gKiB9O1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWlCYWNrQnV0dG9uQ29uZmlnIHtcclxuICAvKiogTGFiZWwgdGV4dCBmb3IgdGhlIGJhY2sgYnV0dG9uLiBAZGVmYXVsdCAnSW5kaWV0cm8nICovXHJcbiAgbGFiZWw/OiBzdHJpbmc7XHJcbiAgLyoqIEx1Y2lkZSBpY29uIG5hbWUuIEBkZWZhdWx0ICdhcnJvdy1sZWZ0JyAqL1xyXG4gIGljb24/OiBzdHJpbmc7XHJcbiAgLyoqIFZpc3VhbCB2YXJpYW50LiBAZGVmYXVsdCAnb3V0bGluZScgKi9cclxuICB2YXJpYW50Pzogc3RyaW5nO1xyXG59XHJcbiJdfQ==
@@ -0,0 +1,14 @@
1
+ /**
2
+ * ng-ui-system — Base Layout entry point.
3
+ *
4
+ * @example
5
+ * ```typescript
6
+ * import {
7
+ * UiBaseLayoutComponent,
8
+ * UiBackButtonConfig,
9
+ * } from 'ng-ui-system';
10
+ * ```
11
+ */
12
+ // Components
13
+ export { UiBaseLayoutComponent } from './base-layout.component';
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCxhQUFhO0FBQ2IsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogbmctdWktc3lzdGVtIOKAlCBCYXNlIExheW91dCBlbnRyeSBwb2ludC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBpbXBvcnQge1xyXG4gKiAgIFVpQmFzZUxheW91dENvbXBvbmVudCxcclxuICogICBVaUJhY2tCdXR0b25Db25maWcsXHJcbiAqIH0gZnJvbSAnbmctdWktc3lzdGVtJztcclxuICogYGBgXHJcbiAqL1xyXG5cclxuLy8gQ29tcG9uZW50c1xyXG5leHBvcnQgeyBVaUJhc2VMYXlvdXRDb21wb25lbnQgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LmNvbXBvbmVudCc7XHJcblxyXG4vLyBUeXBlc1xyXG5leHBvcnQgeyBVaUJhY2tCdXR0b25Db25maWcgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LnR5cGVzJztcclxuIl19
@@ -0,0 +1,196 @@
1
+ import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';
2
+ import { Router } from '@angular/router';
3
+ import { UiButtonComponent } from './button.component';
4
+ import * as i0 from "@angular/core";
5
+ /**
6
+ * Standalone button-group component driven by a `UiButtonDescriptor[]` configuration.
7
+ *
8
+ * Renders a group of `UiButton` instances from declarative descriptors.
9
+ * Supports per-button loading/disabled states, dynamic visibility,
10
+ * alignment control, and optional navigation via Angular Router.
11
+ *
12
+ * @selector ui-button-area
13
+ *
14
+ * @example
15
+ * ```html
16
+ * <ui-button-area
17
+ * [buttons]="actions"
18
+ * align="end"
19
+ * gap="sm"
20
+ * [loadingIds]="currentlyLoading"
21
+ * />
22
+ * ```
23
+ *
24
+ * @example
25
+ * ```typescript
26
+ * // Dynamic configuration
27
+ * actions: UiButtonDescriptor[] = [
28
+ * { id: 'export', label: 'Export Excel', icon: 'file-spreadsheet', variant: 'primary',
29
+ * action: () => this.exportExcel() },
30
+ * { id: 'delete', label: 'Delete', icon: 'trash-2', variant: 'warn',
31
+ * action: () => this.deleteSelected(), hidden: !this.hasSelection },
32
+ * ];
33
+ *
34
+ * // Set loading state from outside:
35
+ * currentlyLoading: string | string[] | null = 'export';
36
+ * ```
37
+ */
38
+ export class UiButtonAreaComponent {
39
+ constructor() {
40
+ /**
41
+ * Array of button descriptors to render.
42
+ * Buttons with `hidden: true` are filtered out.
43
+ */
44
+ this.buttons = [];
45
+ /** Horizontal alignment of the button group. */
46
+ this.align = 'end';
47
+ /** Accessible group label for screen readers. */
48
+ this.ariaLabel = 'Actions';
49
+ /** Gap between buttons (maps to design token spacing). */
50
+ this.gap = 'sm';
51
+ /** Stack buttons vertically on mobile viewports (<600px). */
52
+ this.stackOnMobile = true;
53
+ /**
54
+ * When `true`, all buttons are disabled while any button is in a loading state.
55
+ * Useful to prevent double-actions during async operations.
56
+ */
57
+ this.disableWhileLoading = false;
58
+ /** @internal */
59
+ this._loadingIds = new Set();
60
+ /** @internal Optional router for href navigation. */
61
+ this.router = inject(Router, { optional: true });
62
+ }
63
+ /**
64
+ * Set one or more buttons to a loading state by their `id`.
65
+ * Accepts a single string, an array of strings, or `null` to clear.
66
+ *
67
+ * @example
68
+ * ```html
69
+ * <!-- Single -->
70
+ * <ui-button-area [loadingIds]="'save'" />
71
+ *
72
+ * <!-- Multiple -->
73
+ * <ui-button-area [loadingIds]="['save', 'export']" />
74
+ * ```
75
+ */
76
+ set loadingIds(value) {
77
+ if (value === null || value === undefined) {
78
+ this._loadingIds = new Set();
79
+ }
80
+ else if (Array.isArray(value)) {
81
+ this._loadingIds = new Set(value);
82
+ }
83
+ else {
84
+ this._loadingIds = new Set([value]);
85
+ }
86
+ }
87
+ /** Buttons filtered to visible (non-hidden) entries. */
88
+ get visibleButtons() {
89
+ return this.buttons.filter((b) => !b.hidden);
90
+ }
91
+ /** Assembled CSS class string for the button area container. */
92
+ get areaClasses() {
93
+ return [
94
+ 'ui-button-area',
95
+ `ui-button-area--align-${this.align}`,
96
+ `ui-button-area--gap-${this.gap}`,
97
+ this.stackOnMobile ? 'ui-button-area--stack-mobile' : '',
98
+ ]
99
+ .filter(Boolean)
100
+ .join(' ');
101
+ }
102
+ /** @internal Track function for @for loop. */
103
+ trackButton(index, button) {
104
+ return button.id ?? `idx-${index}`;
105
+ }
106
+ /** Whether a specific button is in a loading state. */
107
+ isButtonLoading(button) {
108
+ return !!button.loading || (!!button.id && this._loadingIds.has(button.id));
109
+ }
110
+ /** Whether a specific button is disabled (includes loading logic). */
111
+ isButtonDisabled(button) {
112
+ if (button.disabled)
113
+ return true;
114
+ if (this.isButtonLoading(button))
115
+ return true;
116
+ if (this.disableWhileLoading && this._loadingIds.size > 0)
117
+ return true;
118
+ return false;
119
+ }
120
+ /** @internal Handles button click: calls action callback and/or navigates. */
121
+ handleButtonClick(button, event) {
122
+ if (button.action) {
123
+ button.action(event);
124
+ }
125
+ if (button.href && this.router) {
126
+ this.router.navigateByUrl(button.href);
127
+ }
128
+ }
129
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiButtonAreaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
130
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiButtonAreaComponent, isStandalone: true, selector: "ui-button-area", inputs: { buttons: "buttons", align: "align", ariaLabel: "ariaLabel", gap: "gap", stackOnMobile: "stackOnMobile", disableWhileLoading: "disableWhileLoading", loadingIds: "loadingIds" }, host: { classAttribute: "ui-button-area-host" }, ngImport: i0, template: `
131
+ <div
132
+ [class]="areaClasses"
133
+ role="group"
134
+ [attr.aria-label]="ariaLabel"
135
+ >
136
+ @for (button of visibleButtons; track trackButton($index, button)) {
137
+ <ui-button
138
+ [attr.data-button-id]="button.id || null"
139
+ [label]="button.label"
140
+ [tooltip]="button.tooltip ?? ''"
141
+ [icon]="button.icon"
142
+ [iconPosition]="button.iconPosition ?? 'trailing'"
143
+ [variant]="button.variant ?? 'primary'"
144
+ [size]="button.size ?? 'md'"
145
+ [loading]="isButtonLoading(button)"
146
+ [disabled]="isButtonDisabled(button)"
147
+ [customClass]="button.customClass"
148
+ (click)="handleButtonClick(button, $event)"
149
+ />
150
+ }
151
+ </div>
152
+ `, isInline: true, styles: [".ui-button-area-host{display:block}@media (max-width: 767.98px){.ui-button-area-host{width:100%}}.ui-button-area{display:flex;flex-wrap:wrap;align-items:center}.ui-button-area--gap-xs{gap:var(--ui-spacing-1)}.ui-button-area--gap-sm{gap:var(--ui-spacing-2)}.ui-button-area--gap-md{gap:var(--ui-spacing-3)}.ui-button-area--gap-lg{gap:var(--ui-spacing-4)}.ui-button-area--gap-xl{gap:var(--ui-spacing-5)}.ui-button-area--align-start{justify-content:flex-start}.ui-button-area--align-center{justify-content:center}.ui-button-area--align-end{justify-content:flex-end}.ui-button-area--align-between{justify-content:space-between}@media (max-width: 767.98px){.ui-button-area--stack-mobile{flex-direction:column;width:100%}.ui-button-area--stack-mobile .ui-button-host{display:flex;width:100%}.ui-button-area--stack-mobile .ui-button{width:100%}}\n"], dependencies: [{ kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
153
+ }
154
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiButtonAreaComponent, decorators: [{
155
+ type: Component,
156
+ args: [{ selector: 'ui-button-area', standalone: true, imports: [UiButtonComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
157
+ class: 'ui-button-area-host',
158
+ }, template: `
159
+ <div
160
+ [class]="areaClasses"
161
+ role="group"
162
+ [attr.aria-label]="ariaLabel"
163
+ >
164
+ @for (button of visibleButtons; track trackButton($index, button)) {
165
+ <ui-button
166
+ [attr.data-button-id]="button.id || null"
167
+ [label]="button.label"
168
+ [tooltip]="button.tooltip ?? ''"
169
+ [icon]="button.icon"
170
+ [iconPosition]="button.iconPosition ?? 'trailing'"
171
+ [variant]="button.variant ?? 'primary'"
172
+ [size]="button.size ?? 'md'"
173
+ [loading]="isButtonLoading(button)"
174
+ [disabled]="isButtonDisabled(button)"
175
+ [customClass]="button.customClass"
176
+ (click)="handleButtonClick(button, $event)"
177
+ />
178
+ }
179
+ </div>
180
+ `, styles: [".ui-button-area-host{display:block}@media (max-width: 767.98px){.ui-button-area-host{width:100%}}.ui-button-area{display:flex;flex-wrap:wrap;align-items:center}.ui-button-area--gap-xs{gap:var(--ui-spacing-1)}.ui-button-area--gap-sm{gap:var(--ui-spacing-2)}.ui-button-area--gap-md{gap:var(--ui-spacing-3)}.ui-button-area--gap-lg{gap:var(--ui-spacing-4)}.ui-button-area--gap-xl{gap:var(--ui-spacing-5)}.ui-button-area--align-start{justify-content:flex-start}.ui-button-area--align-center{justify-content:center}.ui-button-area--align-end{justify-content:flex-end}.ui-button-area--align-between{justify-content:space-between}@media (max-width: 767.98px){.ui-button-area--stack-mobile{flex-direction:column;width:100%}.ui-button-area--stack-mobile .ui-button-host{display:flex;width:100%}.ui-button-area--stack-mobile .ui-button{width:100%}}\n"] }]
181
+ }], propDecorators: { buttons: [{
182
+ type: Input
183
+ }], align: [{
184
+ type: Input
185
+ }], ariaLabel: [{
186
+ type: Input
187
+ }], gap: [{
188
+ type: Input
189
+ }], stackOnMobile: [{
190
+ type: Input
191
+ }], disableWhileLoading: [{
192
+ type: Input
193
+ }], loadingIds: [{
194
+ type: Input
195
+ }] } });
196
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-area.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/button/button-area.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';\r\nimport { Router } from '@angular/router';\r\nimport { UiSize } from '../../core/types';\r\nimport { UiButtonComponent } from './button.component';\r\nimport { UiButtonDescriptor, UiButtonAreaAlign } from './button.types';\r\n\r\n/**\r\n * Standalone button-group component driven by a `UiButtonDescriptor[]` configuration.\r\n *\r\n * Renders a group of `UiButton` instances from declarative descriptors.\r\n * Supports per-button loading/disabled states, dynamic visibility,\r\n * alignment control, and optional navigation via Angular Router.\r\n *\r\n * @selector ui-button-area\r\n *\r\n * @example\r\n * ```html\r\n * <ui-button-area\r\n *   [buttons]=\"actions\"\r\n *   align=\"end\"\r\n *   gap=\"sm\"\r\n *   [loadingIds]=\"currentlyLoading\"\r\n * />\r\n * ```\r\n *\r\n * @example\r\n * ```typescript\r\n * // Dynamic configuration\r\n * actions: UiButtonDescriptor[] = [\r\n *   { id: 'export', label: 'Export Excel', icon: 'file-spreadsheet', variant: 'primary',\r\n *     action: () => this.exportExcel() },\r\n *   { id: 'delete', label: 'Delete', icon: 'trash-2', variant: 'warn',\r\n *     action: () => this.deleteSelected(), hidden: !this.hasSelection },\r\n * ];\r\n *\r\n * // Set loading state from outside:\r\n * currentlyLoading: string | string[] | null = 'export';\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-button-area',\r\n  standalone: true,\r\n  imports: [UiButtonComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-button-area-host',\r\n  },\r\n  template: `\r\n    <div\r\n      [class]=\"areaClasses\"\r\n      role=\"group\"\r\n      [attr.aria-label]=\"ariaLabel\"\r\n    >\r\n      @for (button of visibleButtons; track trackButton($index, button)) {\r\n        <ui-button\r\n          [attr.data-button-id]=\"button.id || null\"\r\n          [label]=\"button.label\"\r\n          [tooltip]=\"button.tooltip ?? ''\"\r\n          [icon]=\"button.icon\"\r\n          [iconPosition]=\"button.iconPosition ?? 'trailing'\"\r\n          [variant]=\"button.variant ?? 'primary'\"\r\n          [size]=\"button.size ?? 'md'\"\r\n          [loading]=\"isButtonLoading(button)\"\r\n          [disabled]=\"isButtonDisabled(button)\"\r\n          [customClass]=\"button.customClass\"\r\n          (click)=\"handleButtonClick(button, $event)\"\r\n        />\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './button-area.component.scss',\r\n})\r\nexport class UiButtonAreaComponent {\r\n  /**\r\n   * Array of button descriptors to render.\r\n   * Buttons with `hidden: true` are filtered out.\r\n   */\r\n  @Input() buttons: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the button group. */\r\n  @Input() align: UiButtonAreaAlign = 'end';\r\n\r\n  /** Accessible group label for screen readers. */\r\n  @Input() ariaLabel = 'Actions';\r\n\r\n  /** Gap between buttons (maps to design token spacing). */\r\n  @Input() gap: UiSize = 'sm';\r\n\r\n  /** Stack buttons vertically on mobile viewports (<600px). */\r\n  @Input() stackOnMobile = true;\r\n\r\n  /**\r\n   * When `true`, all buttons are disabled while any button is in a loading state.\r\n   * Useful to prevent double-actions during async operations.\r\n   */\r\n  @Input() disableWhileLoading = false;\r\n\r\n  /**\r\n   * Set one or more buttons to a loading state by their `id`.\r\n   * Accepts a single string, an array of strings, or `null` to clear.\r\n   *\r\n   * @example\r\n   * ```html\r\n   * <!-- Single -->\r\n   * <ui-button-area [loadingIds]=\"'save'\" />\r\n   *\r\n   * <!-- Multiple -->\r\n   * <ui-button-area [loadingIds]=\"['save', 'export']\" />\r\n   * ```\r\n   */\r\n  @Input()\r\n  set loadingIds(value: string | string[] | null | undefined) {\r\n    if (value === null || value === undefined) {\r\n      this._loadingIds = new Set();\r\n    } else if (Array.isArray(value)) {\r\n      this._loadingIds = new Set(value);\r\n    } else {\r\n      this._loadingIds = new Set([value]);\r\n    }\r\n  }\r\n\r\n  /** @internal */\r\n  private _loadingIds = new Set<string>();\r\n\r\n  /** @internal Optional router for href navigation. */\r\n  private router = inject(Router, { optional: true });\r\n\r\n  /** Buttons filtered to visible (non-hidden) entries. */\r\n  get visibleButtons(): UiButtonDescriptor[] {\r\n    return this.buttons.filter((b) => !b.hidden);\r\n  }\r\n\r\n  /** Assembled CSS class string for the button area container. */\r\n  get areaClasses(): string {\r\n    return [\r\n      'ui-button-area',\r\n      `ui-button-area--align-${this.align}`,\r\n      `ui-button-area--gap-${this.gap}`,\r\n      this.stackOnMobile ? 'ui-button-area--stack-mobile' : '',\r\n    ]\r\n      .filter(Boolean)\r\n      .join(' ');\r\n  }\r\n\r\n  /** @internal Track function for @for loop. */\r\n  trackButton(index: number, button: UiButtonDescriptor): string {\r\n    return button.id ?? `idx-${index}`;\r\n  }\r\n\r\n  /** Whether a specific button is in a loading state. */\r\n  isButtonLoading(button: UiButtonDescriptor): boolean {\r\n    return !!button.loading || (!!button.id && this._loadingIds.has(button.id));\r\n  }\r\n\r\n  /** Whether a specific button is disabled (includes loading logic). */\r\n  isButtonDisabled(button: UiButtonDescriptor): boolean {\r\n    if (button.disabled) return true;\r\n    if (this.isButtonLoading(button)) return true;\r\n    if (this.disableWhileLoading && this._loadingIds.size > 0) return true;\r\n    return false;\r\n  }\r\n\r\n  /** @internal Handles button click: calls action callback and/or navigates. */\r\n  handleButtonClick(button: UiButtonDescriptor, event: MouseEvent): void {\r\n    if (button.action) {\r\n      button.action(event);\r\n    }\r\n    if (button.href && this.router) {\r\n      this.router.navigateByUrl(button.href);\r\n    }\r\n  }\r\n}\r\n"]}