@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,1730 @@
1
+ /**
2
+ * Schema-driven application shell that materializes a full responsive layout
3
+ * from a single `UiLayoutSchema` descriptor.
4
+ *
5
+ * Features:
6
+ * - Two layout modes: sidebar (collapsible) and topbar (multi-bar stack)
7
+ * - Programmatic mode switching with loader masking
8
+ * - Content type: fluid, boxed, fullscreen
9
+ * - Speed-dial FAB with context-sensitive actions
10
+ * - Overlay loader with body-scroll locking
11
+ * - Auto-derived breadcrumbs from navigation schema
12
+ * - Skip-to-content link and full keyboard navigation
13
+ *
14
+ * @selector ui-layout-builder
15
+ *
16
+ * @example
17
+ * ```html
18
+ * <ui-layout-builder [schema]="layoutSchema" />
19
+ * ```
20
+ */
21
+ import { Component, Input, ChangeDetectionStrategy, ChangeDetectorRef, ViewEncapsulation, inject, HostListener, } from '@angular/core';
22
+ import { Router, RouterOutlet, RouterLink, RouterLinkActive, NavigationEnd } from '@angular/router';
23
+ import { NgTemplateOutlet } from '@angular/common';
24
+ import { LucideAngularModule } from 'lucide-angular';
25
+ import { filter } from 'rxjs';
26
+ import { UiLayoutService } from './layout.service';
27
+ import * as i0 from "@angular/core";
28
+ import * as i1 from "lucide-angular";
29
+ export class UiLayoutBuilderComponent {
30
+ constructor() {
31
+ this.layoutService = inject(UiLayoutService);
32
+ this.router = inject(Router);
33
+ this.cdr = inject(ChangeDetectorRef);
34
+ this.subs = [];
35
+ this.currentUrl = '';
36
+ this.expandedIds = new Set();
37
+ this.fabOpen = false;
38
+ this.currentFabConfig = null;
39
+ this.loading = false;
40
+ this.sidebarOpen = false;
41
+ this.sidebarCollapsed = false;
42
+ this.layoutMode = 'sidebar';
43
+ this.contentType = 'fluid';
44
+ this.contentTransitioning = false;
45
+ this.breadcrumbs = [];
46
+ this.userDropdownOpen = false;
47
+ this.hnavDropdownId = null;
48
+ this.navBadges = new Map();
49
+ this.navVisibility = new Map();
50
+ this.barVisibility = new Map();
51
+ }
52
+ /** Resolved page title from the last breadcrumb. */
53
+ get pageTitle() {
54
+ if (!this.breadcrumbs.length)
55
+ return '';
56
+ return this.breadcrumbs[this.breadcrumbs.length - 1].label;
57
+ }
58
+ /** Items flagged for the mobile bottom navigation bar (max 5). */
59
+ get bottomNavItems() {
60
+ return this.schema.navigation.items
61
+ .filter((i) => i.bottomNav && !this.isItemHidden(i) && i.route)
62
+ .sort((a, b) => (a.bottomNavOrder ?? 99) - (b.bottomNavOrder ?? 99))
63
+ .slice(0, 5);
64
+ }
65
+ get mobileMode() {
66
+ return this.schema.navigation.mobileMode || 'both';
67
+ }
68
+ /** Topbar bars filtered by visibility (schema default + service overrides). */
69
+ get visibleBars() {
70
+ if (!this.schema.topbar?.bars)
71
+ return [];
72
+ return this.schema.topbar.bars.filter((bar) => this.layoutService.isBarVisible(bar.id, bar.visible !== false));
73
+ }
74
+ ngOnInit() {
75
+ this.currentUrl = this.router.url;
76
+ this.autoExpandFromUrl(this.currentUrl);
77
+ if (this.schema.mode) {
78
+ this.layoutService._setInitialLayoutMode(this.schema.mode);
79
+ }
80
+ if (this.schema.contentType) {
81
+ this.layoutService._setInitialContentType(this.schema.contentType);
82
+ }
83
+ if (this.schema.fab) {
84
+ this.layoutService._setDefaultFabConfig(this.schema.fab);
85
+ }
86
+ if (this.schema.loader?.showOnInit) {
87
+ this.layoutService.showLoader();
88
+ }
89
+ this.breadcrumbs = this.computeBreadcrumbs(this.currentUrl);
90
+ this.subs.push(this.layoutService.loading$.subscribe((v) => {
91
+ this.loading = v;
92
+ this.cdr.markForCheck();
93
+ }), this.layoutService.sidebarOpen$.subscribe((v) => {
94
+ this.sidebarOpen = v;
95
+ this.cdr.markForCheck();
96
+ }), this.layoutService.sidebarCollapsed$.subscribe((v) => {
97
+ this.sidebarCollapsed = v;
98
+ this.cdr.markForCheck();
99
+ }), this.layoutService.fabConfig$.subscribe((config) => {
100
+ this.currentFabConfig = config;
101
+ this.cdr.markForCheck();
102
+ }), this.layoutService.navBadges$.subscribe((m) => {
103
+ this.navBadges = m;
104
+ this.cdr.markForCheck();
105
+ }), this.layoutService.navVisibility$.subscribe((m) => {
106
+ this.navVisibility = m;
107
+ this.cdr.markForCheck();
108
+ }), this.layoutService.layoutMode$.subscribe((mode) => {
109
+ this.layoutMode = mode;
110
+ this.cdr.markForCheck();
111
+ }), this.layoutService.contentType$.subscribe((type) => {
112
+ this.contentType = type;
113
+ this.cdr.markForCheck();
114
+ }), this.layoutService.barVisibility$.subscribe((m) => {
115
+ this.barVisibility = m;
116
+ this.cdr.markForCheck();
117
+ }), this.router.events
118
+ .pipe(filter((e) => e instanceof NavigationEnd))
119
+ .subscribe((e) => {
120
+ this.currentUrl = e.urlAfterRedirects;
121
+ this.autoExpandFromUrl(this.currentUrl);
122
+ this.layoutService.closeMobileSidebar();
123
+ this.breadcrumbs = this.computeBreadcrumbs(this.currentUrl);
124
+ this.contentTransitioning = true;
125
+ this.userDropdownOpen = false;
126
+ this.hnavDropdownId = null;
127
+ if (this.schema.loader?.showOnInit && this.layoutService.isLoading()) {
128
+ const hide = () => this.layoutService.hideLoader();
129
+ const minDuration = this.schema.loader.minDuration ?? 0;
130
+ if (minDuration > 0) {
131
+ setTimeout(hide, minDuration);
132
+ }
133
+ else {
134
+ hide();
135
+ }
136
+ }
137
+ this.cdr.markForCheck();
138
+ }));
139
+ }
140
+ ngOnDestroy() {
141
+ this.subs.forEach((s) => s.unsubscribe());
142
+ }
143
+ onEscape() {
144
+ if (this.fabOpen) {
145
+ this.fabOpen = false;
146
+ }
147
+ if (this.userDropdownOpen) {
148
+ this.userDropdownOpen = false;
149
+ }
150
+ if (this.hnavDropdownId) {
151
+ this.hnavDropdownId = null;
152
+ }
153
+ if (this.sidebarOpen) {
154
+ this.layoutService.closeMobileSidebar();
155
+ }
156
+ }
157
+ onDocumentClick(event) {
158
+ const target = event.target;
159
+ if (this.userDropdownOpen && !target.closest('.ui-layout__user-dropdown')) {
160
+ this.userDropdownOpen = false;
161
+ this.cdr.markForCheck();
162
+ }
163
+ if (this.hnavDropdownId && !target.closest('.ui-layout__hnav-dropdown')) {
164
+ this.hnavDropdownId = null;
165
+ this.cdr.markForCheck();
166
+ }
167
+ }
168
+ // ── Navigation helpers ──────────────────────────────────────────
169
+ getItemsForSection(section) {
170
+ const idSet = new Set(section.itemIds);
171
+ return this.schema.navigation.items.filter((i) => idSet.has(i.id));
172
+ }
173
+ isItemHidden(item) {
174
+ const override = this.navVisibility.get(item.id);
175
+ if (override !== undefined)
176
+ return !override;
177
+ return item.hidden ?? false;
178
+ }
179
+ getItemBadge(item) {
180
+ return this.navBadges.get(item.id) ?? item.badge ?? null;
181
+ }
182
+ isParentActive(item) {
183
+ if (!item.route)
184
+ return false;
185
+ return this.currentUrl === item.route || this.currentUrl.startsWith(item.route + '/');
186
+ }
187
+ isRouteActive(item) {
188
+ if (!item.route)
189
+ return false;
190
+ if (item.routeActiveExact)
191
+ return this.currentUrl === item.route;
192
+ return this.currentUrl === item.route || this.currentUrl.startsWith(item.route + '/');
193
+ }
194
+ toggleExpand(item) {
195
+ if (this.expandedIds.has(item.id)) {
196
+ this.expandedIds.delete(item.id);
197
+ }
198
+ else {
199
+ this.expandedIds.add(item.id);
200
+ if (!this.isParentActive(item) && item.children?.length) {
201
+ const firstChild = item.children[0]?.items[0];
202
+ if (firstChild?.route) {
203
+ this.router.navigateByUrl(firstChild.route);
204
+ }
205
+ }
206
+ }
207
+ }
208
+ onNavClick() {
209
+ this.layoutService.closeMobileSidebar();
210
+ }
211
+ // ── FAB ─────────────────────────────────────────────────────────
212
+ onFabMainClick() {
213
+ if (this.currentFabConfig?.secondaryActions?.length) {
214
+ this.fabOpen = !this.fabOpen;
215
+ }
216
+ else {
217
+ this.currentFabConfig?.mainAction.action?.();
218
+ }
219
+ }
220
+ onFabAction(action) {
221
+ this.fabOpen = false;
222
+ action.action?.();
223
+ }
224
+ // ── Topbar helpers ──────────────────────────────────────────────
225
+ getTopbarNavItems(bar) {
226
+ return bar.navigation?.items ?? this.schema.navigation.items;
227
+ }
228
+ toggleHnavDropdown(itemId) {
229
+ this.hnavDropdownId = this.hnavDropdownId === itemId ? null : itemId;
230
+ }
231
+ dismissBar(barId) {
232
+ this.layoutService.hideBar(barId);
233
+ }
234
+ onDropdownAction(item) {
235
+ this.userDropdownOpen = false;
236
+ item.action?.();
237
+ }
238
+ // ── Breadcrumb derivation from nav schema ───────────────────────
239
+ /**
240
+ * Derives breadcrumbs by matching the current URL against the
241
+ * navigation schema. Walks top-level items, sections, and children
242
+ * to build a trail from root to current page.
243
+ */
244
+ computeBreadcrumbs(url) {
245
+ const items = this.schema.navigation.items;
246
+ const sections = this.schema.navigation.sections;
247
+ for (const item of items) {
248
+ if (item.type === 'divider' || item.type === 'external')
249
+ continue;
250
+ if (item.children?.length) {
251
+ for (const group of item.children) {
252
+ for (const child of group.items) {
253
+ if (child.route && this.urlMatches(url, child.route, child.routeActiveExact)) {
254
+ const trail = [];
255
+ const sectionLabel = this.findSectionLabel(item.id, sections);
256
+ if (sectionLabel) {
257
+ trail.push({ label: sectionLabel, isLast: false });
258
+ }
259
+ trail.push({ label: item.label, url: item.route, isLast: false });
260
+ trail.push({ label: child.label, isLast: true });
261
+ return trail;
262
+ }
263
+ }
264
+ }
265
+ }
266
+ if (item.route && this.urlMatches(url, item.route, item.routeActiveExact)) {
267
+ const trail = [];
268
+ const sectionLabel = this.findSectionLabel(item.id, sections);
269
+ if (sectionLabel) {
270
+ trail.push({ label: sectionLabel, isLast: false });
271
+ }
272
+ trail.push({ label: item.label, isLast: true });
273
+ return trail;
274
+ }
275
+ }
276
+ return [];
277
+ }
278
+ urlMatches(current, route, exact) {
279
+ if (exact)
280
+ return current === route;
281
+ return current === route || current.startsWith(route + '/');
282
+ }
283
+ findSectionLabel(itemId, sections) {
284
+ if (!sections?.length)
285
+ return null;
286
+ const section = sections.find((s) => s.itemIds.includes(itemId));
287
+ return section?.label ?? null;
288
+ }
289
+ autoExpandFromUrl(url) {
290
+ for (const item of this.schema.navigation.items) {
291
+ if (item.children && item.route && (url === item.route || url.startsWith(item.route + '/'))) {
292
+ this.expandedIds.add(item.id);
293
+ }
294
+ }
295
+ }
296
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiLayoutBuilderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
297
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiLayoutBuilderComponent, isStandalone: true, selector: "ui-layout-builder", inputs: { schema: "schema" }, host: { listeners: { "document:keydown.escape": "onEscape()", "document:click": "onDocumentClick($event)" }, classAttribute: "ui-layout-builder-host" }, ngImport: i0, template: `
298
+ <!-- Skip to content (a11y) -->
299
+ <a class="ui-layout__skip-link" href="#ui-layout-main">
300
+ Vai al contenuto principale
301
+ </a>
302
+
303
+ <div
304
+ class="ui-layout"
305
+ [class.ui-layout--sidebar-collapsed]="layoutMode === 'sidebar' && sidebarCollapsed"
306
+ [class.ui-layout--topbar]="layoutMode === 'topbar'"
307
+ >
308
+ <!-- ─── Overlay Loader ─────────────────────────────────────── -->
309
+ @if (loading) {
310
+ <div
311
+ class="ui-layout__loader-overlay"
312
+ role="alert"
313
+ aria-live="assertive"
314
+ aria-label="Caricamento in corso"
315
+ >
316
+ <div class="ui-layout__loader-spinner" aria-hidden="true">
317
+ <svg viewBox="0 0 50 50" class="ui-layout__loader-svg">
318
+ <circle cx="25" cy="25" r="20" fill="none" stroke-width="4" />
319
+ </svg>
320
+ </div>
321
+ </div>
322
+ }
323
+
324
+ @switch (layoutMode) {
325
+ <!-- ════════════════════════════════════════════════════════ -->
326
+ <!-- ═══ SIDEBAR MODE ═════════════════════════════════════ -->
327
+ <!-- ════════════════════════════════════════════════════════ -->
328
+ @case ('sidebar') {
329
+ <!-- Mobile Top Bar -->
330
+ <header class="ui-layout__topbar">
331
+ <button
332
+ class="ui-layout__hamburger"
333
+ (click)="layoutService.openMobileSidebar()"
334
+ aria-label="Apri navigazione"
335
+ >
336
+ <span class="ui-layout__hamburger-line"></span>
337
+ <span class="ui-layout__hamburger-line"></span>
338
+ <span class="ui-layout__hamburger-line"></span>
339
+ </button>
340
+ @if (schema.header.logo) {
341
+ @if (schema.header.logo.src) {
342
+ <img
343
+ class="ui-layout__topbar-logo-img"
344
+ [src]="schema.header.logo.src"
345
+ [alt]="schema.header.logo.alt || schema.header.title"
346
+ />
347
+ } @else if (schema.header.logo.text) {
348
+ <span class="ui-layout__topbar-logo">{{ schema.header.logo.text }}</span>
349
+ }
350
+ }
351
+ <span class="ui-layout__topbar-title">{{ schema.header.title }}</span>
352
+ </header>
353
+
354
+ <!-- Mobile Backdrop -->
355
+ @if (sidebarOpen) {
356
+ <div
357
+ class="ui-layout__backdrop"
358
+ (click)="layoutService.closeMobileSidebar()"
359
+ aria-hidden="true"
360
+ ></div>
361
+ }
362
+
363
+ <!-- Body (sidebar + content) -->
364
+ <div class="ui-layout__body">
365
+ <aside
366
+ class="ui-layout__sidebar"
367
+ [class.ui-layout__sidebar--collapsed]="sidebarCollapsed"
368
+ [class.ui-layout__sidebar--open]="sidebarOpen"
369
+ role="navigation"
370
+ aria-label="Navigazione principale"
371
+ >
372
+ <div class="ui-layout__sidebar-header">
373
+ <div class="ui-layout__sidebar-brand">
374
+ @if (schema.header.logo) {
375
+ @if (schema.header.logo.src) {
376
+ <img
377
+ class="ui-layout__sidebar-logo-img"
378
+ [src]="schema.header.logo.src"
379
+ [alt]="schema.header.logo.alt || schema.header.title"
380
+ />
381
+ } @else if (schema.header.logo.text) {
382
+ <span class="ui-layout__sidebar-logo">{{ schema.header.logo.text }}</span>
383
+ }
384
+ }
385
+ @if (!sidebarCollapsed || sidebarOpen) {
386
+ <span class="ui-layout__sidebar-title">{{ schema.header.title }}</span>
387
+ }
388
+ </div>
389
+
390
+ @if (schema.navigation.collapsible !== false) {
391
+ <button
392
+ class="ui-layout__collapse-btn ui-layout__desktop-only"
393
+ (click)="layoutService.toggleSidebar()"
394
+ [attr.aria-label]="sidebarCollapsed ? 'Espandi sidebar' : 'Comprimi sidebar'"
395
+ >
396
+ <lucide-icon
397
+ name="chevrons-left"
398
+ [size]="16"
399
+ class="ui-layout__collapse-icon"
400
+ aria-hidden="true"
401
+ />
402
+ </button>
403
+ }
404
+
405
+ <button
406
+ class="ui-layout__close-btn ui-layout__mobile-only"
407
+ (click)="layoutService.closeMobileSidebar()"
408
+ aria-label="Chiudi navigazione"
409
+ >
410
+ <lucide-icon name="x" [size]="16" aria-hidden="true" />
411
+ </button>
412
+ </div>
413
+
414
+ @if (!sidebarCollapsed || sidebarOpen) {
415
+ <nav class="ui-layout__sidebar-nav">
416
+ @if (schema.navigation.sections?.length) {
417
+ @for (section of schema.navigation.sections; track section.id) {
418
+ <div class="ui-layout__nav-section">
419
+ <span class="ui-layout__nav-section-title">{{ section.label }}</span>
420
+ @for (item of getItemsForSection(section); track item.id) {
421
+ @if (!isItemHidden(item)) {
422
+ <ng-container
423
+ *ngTemplateOutlet="navItemTpl; context: { $implicit: item }"
424
+ />
425
+ }
426
+ }
427
+ </div>
428
+ }
429
+ } @else {
430
+ <div class="ui-layout__nav-section">
431
+ @for (item of schema.navigation.items; track item.id) {
432
+ @if (!isItemHidden(item)) {
433
+ <ng-container
434
+ *ngTemplateOutlet="navItemTpl; context: { $implicit: item }"
435
+ />
436
+ }
437
+ }
438
+ </div>
439
+ }
440
+ </nav>
441
+ }
442
+ </aside>
443
+
444
+ <!-- Main content area -->
445
+ <main
446
+ class="ui-layout__content"
447
+ [class.ui-layout__content--boxed]="contentType === 'boxed'"
448
+ [class.ui-layout__content--fullscreen]="contentType === 'fullscreen'"
449
+ [class.ui-layout__content--transitioning]="contentTransitioning"
450
+ (animationend)="contentTransitioning = false"
451
+ id="ui-layout-main"
452
+ tabindex="-1"
453
+ >
454
+ <ng-container *ngTemplateOutlet="pageHeaderTpl" />
455
+ <router-outlet />
456
+ </main>
457
+ </div>
458
+ }
459
+
460
+ <!-- ════════════════════════════════════════════════════════ -->
461
+ <!-- ═══ TOPBAR MODE ══════════════════════════════════════ -->
462
+ <!-- ════════════════════════════════════════════════════════ -->
463
+ @case ('topbar') {
464
+ <!-- Bar stack -->
465
+ <div class="ui-layout__bar-stack">
466
+ @for (bar of visibleBars; track bar.id) {
467
+ @switch (bar.type) {
468
+ @case ('notification') {
469
+ @if (bar.notification) {
470
+ <div
471
+ class="ui-layout__bar ui-layout__bar--notification"
472
+ [class]="'ui-layout__bar--notification-' + (bar.notification.variant || 'primary')"
473
+ role="alert"
474
+ >
475
+ @if (bar.notification.icon) {
476
+ <lucide-icon [name]="bar.notification.icon" [size]="16" aria-hidden="true" />
477
+ }
478
+ <span class="ui-layout__bar-text">{{ bar.notification.text }}</span>
479
+ @if (bar.notification.dismissible !== false) {
480
+ <button
481
+ class="ui-layout__bar-dismiss"
482
+ (click)="dismissBar(bar.id)"
483
+ aria-label="Chiudi notifica"
484
+ >
485
+ <lucide-icon name="x" [size]="14" aria-hidden="true" />
486
+ </button>
487
+ }
488
+ </div>
489
+ }
490
+ }
491
+
492
+ @case ('brand') {
493
+ @if (bar.brand; as brand) {
494
+ <div class="ui-layout__bar ui-layout__bar--brand">
495
+ <div class="ui-layout__bar-brand-left">
496
+ <button
497
+ class="ui-layout__hamburger ui-layout__topbar-hamburger"
498
+ (click)="layoutService.openMobileSidebar()"
499
+ aria-label="Apri navigazione"
500
+ >
501
+ <span class="ui-layout__hamburger-line"></span>
502
+ <span class="ui-layout__hamburger-line"></span>
503
+ <span class="ui-layout__hamburger-line"></span>
504
+ </button>
505
+ @if (brand.logo; as logo) {
506
+ @if (logo.src) {
507
+ <img
508
+ class="ui-layout__bar-logo-img"
509
+ [src]="logo.src"
510
+ [alt]="logo.alt || brand.title || ''"
511
+ />
512
+ } @else if (logo.text) {
513
+ <span class="ui-layout__bar-logo">{{ logo.text }}</span>
514
+ }
515
+ }
516
+ @if (brand.title; as brandTitle) {
517
+ <span class="ui-layout__bar-title">{{ brandTitle }}</span>
518
+ }
519
+ </div>
520
+ @if (brand.userDropdown; as dropdown) {
521
+ <div class="ui-layout__user-dropdown">
522
+ <button
523
+ class="ui-layout__user-dropdown-trigger"
524
+ (click)="userDropdownOpen = !userDropdownOpen"
525
+ [attr.aria-expanded]="userDropdownOpen"
526
+ aria-haspopup="true"
527
+ >
528
+ @if (dropdown.avatar?.src) {
529
+ <img
530
+ class="ui-layout__user-avatar"
531
+ [src]="dropdown.avatar!.src"
532
+ [alt]="dropdown.avatar!.alt || dropdown.label"
533
+ />
534
+ } @else {
535
+ <lucide-icon
536
+ [name]="dropdown.icon || 'user'"
537
+ [size]="18"
538
+ aria-hidden="true"
539
+ />
540
+ }
541
+ <span class="ui-layout__user-label">{{ dropdown.label }}</span>
542
+ <lucide-icon name="chevron-down" [size]="14" aria-hidden="true" class="ui-layout__user-chevron" />
543
+ </button>
544
+ @if (userDropdownOpen) {
545
+ <div class="ui-layout__user-dropdown-menu" role="menu">
546
+ @for (dItem of dropdown.items; track dItem.id) {
547
+ @if (dItem.divider) {
548
+ <hr class="ui-layout__user-dropdown-divider" />
549
+ }
550
+ @if (dItem.route) {
551
+ <a
552
+ class="ui-layout__user-dropdown-item"
553
+ [routerLink]="dItem.route"
554
+ role="menuitem"
555
+ (click)="userDropdownOpen = false"
556
+ >
557
+ @if (dItem.icon) {
558
+ <lucide-icon [name]="dItem.icon" [size]="16" aria-hidden="true" />
559
+ }
560
+ {{ dItem.label }}
561
+ </a>
562
+ } @else {
563
+ <button
564
+ class="ui-layout__user-dropdown-item"
565
+ role="menuitem"
566
+ (click)="onDropdownAction(dItem)"
567
+ >
568
+ @if (dItem.icon) {
569
+ <lucide-icon [name]="dItem.icon" [size]="16" aria-hidden="true" />
570
+ }
571
+ {{ dItem.label }}
572
+ </button>
573
+ }
574
+ }
575
+ </div>
576
+ }
577
+ </div>
578
+ }
579
+ </div>
580
+ }
581
+ }
582
+
583
+ @case ('navigation') {
584
+ <nav class="ui-layout__bar ui-layout__bar--navigation" role="navigation" aria-label="Navigazione principale">
585
+ @for (item of getTopbarNavItems(bar); track item.id) {
586
+ @if (!isItemHidden(item)) {
587
+ @if (item.children?.length) {
588
+ <div class="ui-layout__hnav-dropdown" [class.ui-layout__hnav-dropdown--open]="hnavDropdownId === item.id">
589
+ <button
590
+ class="ui-layout__hnav-link ui-layout__hnav-link--parent"
591
+ [class.ui-layout__hnav-link--active]="isParentActive(item)"
592
+ (click)="toggleHnavDropdown(item.id)"
593
+ [attr.aria-expanded]="hnavDropdownId === item.id"
594
+ >
595
+ @if (item.icon) {
596
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
597
+ }
598
+ <span>{{ item.label }}</span>
599
+ <lucide-icon name="chevron-down" [size]="14" aria-hidden="true" class="ui-layout__hnav-chevron" />
600
+ </button>
601
+ @if (hnavDropdownId === item.id) {
602
+ <div class="ui-layout__hnav-dropdown-menu">
603
+ @for (group of item.children; track group.label) {
604
+ <span class="ui-layout__hnav-dropdown-label">{{ group.label }}</span>
605
+ @for (child of group.items; track child.id) {
606
+ @if (!isItemHidden(child)) {
607
+ <a
608
+ class="ui-layout__hnav-dropdown-item"
609
+ [routerLink]="child.route"
610
+ routerLinkActive="ui-layout__hnav-dropdown-item--active"
611
+ (click)="hnavDropdownId = null; onNavClick()"
612
+ >
613
+ {{ child.label }}
614
+ </a>
615
+ }
616
+ }
617
+ }
618
+ </div>
619
+ }
620
+ </div>
621
+ } @else if (item.type === 'external') {
622
+ <a
623
+ class="ui-layout__hnav-link"
624
+ [href]="item.href"
625
+ [target]="item.target || '_blank'"
626
+ [attr.rel]="item.target === '_blank' ? 'noopener noreferrer' : null"
627
+ >
628
+ @if (item.icon) {
629
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
630
+ }
631
+ <span>{{ item.label }}</span>
632
+ <lucide-icon name="external-link" [size]="12" aria-hidden="true" />
633
+ </a>
634
+ } @else if (item.type !== 'divider') {
635
+ <a
636
+ class="ui-layout__hnav-link"
637
+ [routerLink]="item.route"
638
+ routerLinkActive="ui-layout__hnav-link--active"
639
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
640
+ >
641
+ @if (item.icon) {
642
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
643
+ }
644
+ <span>{{ item.label }}</span>
645
+ @if (getItemBadge(item)) {
646
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
647
+ }
648
+ </a>
649
+ }
650
+ }
651
+ }
652
+ </nav>
653
+ }
654
+
655
+ @case ('links') {
656
+ @if (bar.links) {
657
+ <div
658
+ class="ui-layout__bar ui-layout__bar--links"
659
+ [style.justify-content]="bar.links.align === 'start' ? 'flex-start' : bar.links.align === 'center' ? 'center' : bar.links.align === 'space-between' ? 'space-between' : 'flex-end'"
660
+ >
661
+ @for (lnk of bar.links.items; track lnk.label || lnk.ariaLabel || $index) {
662
+ @if (lnk.route) {
663
+ <a class="ui-layout__bar-link" [routerLink]="lnk.route" [attr.aria-label]="lnk.ariaLabel || null">
664
+ @if (lnk.icon) {
665
+ <lucide-icon [name]="lnk.icon" [size]="14" aria-hidden="true" />
666
+ }
667
+ @if (lnk.label) {
668
+ <span>{{ lnk.label }}</span>
669
+ }
670
+ </a>
671
+ } @else if (lnk.href) {
672
+ <a
673
+ class="ui-layout__bar-link"
674
+ [href]="lnk.href"
675
+ [target]="lnk.target || '_self'"
676
+ [attr.rel]="lnk.target === '_blank' ? 'noopener noreferrer' : null"
677
+ [attr.aria-label]="lnk.ariaLabel || null"
678
+ >
679
+ @if (lnk.icon) {
680
+ <lucide-icon [name]="lnk.icon" [size]="14" aria-hidden="true" />
681
+ }
682
+ @if (lnk.label) {
683
+ <span>{{ lnk.label }}</span>
684
+ }
685
+ </a>
686
+ }
687
+ }
688
+ </div>
689
+ }
690
+ }
691
+ }
692
+ }
693
+ </div>
694
+
695
+ <!-- Mobile Backdrop (topbar mode) -->
696
+ @if (sidebarOpen) {
697
+ <div
698
+ class="ui-layout__backdrop ui-layout__backdrop--topbar"
699
+ (click)="layoutService.closeMobileSidebar()"
700
+ aria-hidden="true"
701
+ ></div>
702
+ }
703
+
704
+ <!-- Mobile drawer (topbar mode) -->
705
+ <aside
706
+ class="ui-layout__sidebar ui-layout__sidebar--topbar-drawer"
707
+ [class.ui-layout__sidebar--open]="sidebarOpen"
708
+ role="navigation"
709
+ aria-label="Navigazione principale"
710
+ >
711
+ <div class="ui-layout__sidebar-header">
712
+ <div class="ui-layout__sidebar-brand">
713
+ @if (schema.header.logo) {
714
+ @if (schema.header.logo.src) {
715
+ <img class="ui-layout__sidebar-logo-img" [src]="schema.header.logo.src" [alt]="schema.header.logo.alt || schema.header.title" />
716
+ } @else if (schema.header.logo.text) {
717
+ <span class="ui-layout__sidebar-logo">{{ schema.header.logo.text }}</span>
718
+ }
719
+ }
720
+ <span class="ui-layout__sidebar-title">{{ schema.header.title }}</span>
721
+ </div>
722
+ <button class="ui-layout__close-btn" (click)="layoutService.closeMobileSidebar()" aria-label="Chiudi navigazione">
723
+ <lucide-icon name="x" [size]="16" aria-hidden="true" />
724
+ </button>
725
+ </div>
726
+ <nav class="ui-layout__sidebar-nav">
727
+ @if (schema.navigation.sections?.length) {
728
+ @for (section of schema.navigation.sections; track section.id) {
729
+ <div class="ui-layout__nav-section">
730
+ <span class="ui-layout__nav-section-title">{{ section.label }}</span>
731
+ @for (item of getItemsForSection(section); track item.id) {
732
+ @if (!isItemHidden(item)) {
733
+ <ng-container *ngTemplateOutlet="navItemTpl; context: { $implicit: item }" />
734
+ }
735
+ }
736
+ </div>
737
+ }
738
+ } @else {
739
+ <div class="ui-layout__nav-section">
740
+ @for (item of schema.navigation.items; track item.id) {
741
+ @if (!isItemHidden(item)) {
742
+ <ng-container *ngTemplateOutlet="navItemTpl; context: { $implicit: item }" />
743
+ }
744
+ }
745
+ </div>
746
+ }
747
+ </nav>
748
+ </aside>
749
+
750
+ <!-- Main content area (topbar mode) -->
751
+ <main
752
+ class="ui-layout__content ui-layout__content--topbar"
753
+ [class.ui-layout__content--boxed]="contentType === 'boxed'"
754
+ [class.ui-layout__content--fullscreen]="contentType === 'fullscreen'"
755
+ [class.ui-layout__content--transitioning]="contentTransitioning"
756
+ (animationend)="contentTransitioning = false"
757
+ id="ui-layout-main"
758
+ tabindex="-1"
759
+ >
760
+ <ng-container *ngTemplateOutlet="pageHeaderTpl" />
761
+ <router-outlet />
762
+ </main>
763
+ }
764
+ }
765
+
766
+ <!-- ─── Footer ─────────────────────────────────────────────── -->
767
+ @if (schema.footer) {
768
+ <footer class="ui-layout__footer">
769
+ @if (schema.footer.links?.length) {
770
+ <nav class="ui-layout__footer-links" aria-label="Link footer">
771
+ @for (link of schema.footer.links; track link.label) {
772
+ @if (link.route) {
773
+ <a class="ui-layout__footer-link" [routerLink]="link.route">
774
+ @if (link.icon) {
775
+ <lucide-icon [name]="link.icon" [size]="14" aria-hidden="true" />
776
+ }
777
+ {{ link.label }}
778
+ </a>
779
+ } @else if (link.href) {
780
+ <a
781
+ class="ui-layout__footer-link"
782
+ [href]="link.href"
783
+ [target]="link.target || '_self'"
784
+ [attr.rel]="link.target === '_blank' ? 'noopener noreferrer' : null"
785
+ >
786
+ @if (link.icon) {
787
+ <lucide-icon [name]="link.icon" [size]="14" aria-hidden="true" />
788
+ }
789
+ {{ link.label }}
790
+ </a>
791
+ }
792
+ }
793
+ </nav>
794
+ }
795
+ @if (schema.footer.text) {
796
+ <span class="ui-layout__footer-text">{{ schema.footer.text }}</span>
797
+ }
798
+ </footer>
799
+ }
800
+
801
+ <!-- ─── Speed Dial FAB ─────────────────────────────────────── -->
802
+ @if (currentFabConfig) {
803
+ <div
804
+ class="ui-layout__fab"
805
+ [class.ui-layout__fab--open]="fabOpen"
806
+ [class.ui-layout__fab--bottom-left]="currentFabConfig.position === 'bottom-left'"
807
+ role="complementary"
808
+ aria-label="Azioni rapide"
809
+ >
810
+ @if (fabOpen && currentFabConfig.secondaryActions?.length) {
811
+ <div class="ui-layout__fab-actions">
812
+ @for (action of currentFabConfig.secondaryActions; track action.id; let i = $index) {
813
+ <button
814
+ class="ui-layout__fab-action"
815
+ [class]="'ui-layout__fab-action--' + (action.variant || 'neutral')"
816
+ [attr.aria-label]="action.ariaLabel || action.label"
817
+ [title]="action.label"
818
+ [style.animation-delay]="(i * 50) + 'ms'"
819
+ (click)="onFabAction(action)"
820
+ >
821
+ <lucide-icon [name]="action.icon" [size]="18" aria-hidden="true" />
822
+ <span class="ui-layout__fab-action-label">{{ action.label }}</span>
823
+ </button>
824
+ }
825
+ </div>
826
+ }
827
+ <button
828
+ class="ui-layout__fab-main"
829
+ [class]="'ui-layout__fab-main--' + (currentFabConfig.mainAction.variant || 'primary')"
830
+ [attr.aria-label]="currentFabConfig.mainAction.ariaLabel || currentFabConfig.mainAction.label"
831
+ [attr.aria-expanded]="currentFabConfig.secondaryActions?.length ? fabOpen : null"
832
+ (click)="onFabMainClick()"
833
+ >
834
+ <lucide-icon
835
+ [name]="currentFabConfig.mainAction.icon"
836
+ [size]="24"
837
+ aria-hidden="true"
838
+ class="ui-layout__fab-main-icon"
839
+ />
840
+ </button>
841
+ </div>
842
+ }
843
+
844
+ <!-- ─── Bottom Navigation (mobile, sidebar mode) ────────────── -->
845
+ @if (layoutMode === 'sidebar' && bottomNavItems.length > 0 && mobileMode !== 'drawer') {
846
+ <nav
847
+ class="ui-layout__bottom-nav"
848
+ role="navigation"
849
+ aria-label="Navigazione rapida"
850
+ >
851
+ @for (item of bottomNavItems; track item.id) {
852
+ <a
853
+ class="ui-layout__bottom-nav-item"
854
+ [routerLink]="item.route"
855
+ routerLinkActive="ui-layout__bottom-nav-item--active"
856
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
857
+ [attr.aria-current]="isRouteActive(item) ? 'page' : null"
858
+ >
859
+ @if (item.icon) {
860
+ <lucide-icon [name]="item.icon" [size]="20" aria-hidden="true" />
861
+ }
862
+ <span class="ui-layout__bottom-nav-label">{{ item.label }}</span>
863
+ @if (getItemBadge(item)) {
864
+ <span class="ui-layout__badge" aria-label="Notifica">{{ getItemBadge(item) }}</span>
865
+ }
866
+ </a>
867
+ }
868
+ </nav>
869
+ }
870
+ </div>
871
+
872
+ <!-- ═══ SHARED TEMPLATES ═══════════════════════════════════════ -->
873
+
874
+ <!-- Page header with auto-derived breadcrumbs -->
875
+ <ng-template #pageHeaderTpl>
876
+ @if (schema.pageHeader?.show !== false && breadcrumbs.length > 0) {
877
+ <header class="ui-layout__page-header">
878
+ <nav aria-label="Breadcrumb" class="ui-layout__breadcrumb-nav">
879
+ <ol class="ui-layout__breadcrumb" itemscope itemtype="https://schema.org/BreadcrumbList">
880
+ @if (schema.pageHeader?.showHome !== false) {
881
+ <li class="ui-layout__breadcrumb-item" itemprop="itemListElement" itemscope itemtype="https://schema.org/ListItem">
882
+ <a itemprop="item" [routerLink]="schema.pageHeader?.homeRoute || '/'" class="ui-layout__breadcrumb-link" aria-label="Home">
883
+ <lucide-icon name="home" [size]="14" aria-hidden="true" />
884
+ <span itemprop="name" class="ui-layout__breadcrumb-text">Home</span>
885
+ </a>
886
+ <meta itemprop="position" content="1" />
887
+ <span class="ui-layout__breadcrumb-sep" aria-hidden="true">
888
+ <lucide-icon name="chevron-right" [size]="14" />
889
+ </span>
890
+ </li>
891
+ }
892
+ @for (crumb of breadcrumbs; track crumb.label; let i = $index) {
893
+ <li
894
+ class="ui-layout__breadcrumb-item"
895
+ [class.ui-layout__breadcrumb-item--current]="crumb.isLast"
896
+ itemprop="itemListElement"
897
+ itemscope
898
+ itemtype="https://schema.org/ListItem"
899
+ >
900
+ @if (!crumb.isLast && crumb.url) {
901
+ <a itemprop="item" [routerLink]="crumb.url" class="ui-layout__breadcrumb-link">
902
+ <span itemprop="name" class="ui-layout__breadcrumb-text">{{ crumb.label }}</span>
903
+ </a>
904
+ <span class="ui-layout__breadcrumb-sep" aria-hidden="true">
905
+ <lucide-icon name="chevron-right" [size]="14" />
906
+ </span>
907
+ } @else {
908
+ <span itemprop="name" class="ui-layout__breadcrumb-text ui-layout__breadcrumb-text--current" aria-current="page">
909
+ {{ crumb.label }}
910
+ </span>
911
+ }
912
+ <meta itemprop="position" [attr.content]="(schema.pageHeader?.showHome !== false) ? i + 2 : i + 1" />
913
+ </li>
914
+ }
915
+ </ol>
916
+ </nav>
917
+ <h1 class="ui-layout__page-title">{{ pageTitle }}</h1>
918
+ </header>
919
+ }
920
+ </ng-template>
921
+
922
+ <!-- Nav item template (sidebar, recursive-friendly) -->
923
+ <ng-template #navItemTpl let-item>
924
+ @if (item.type === 'divider') {
925
+ <hr class="ui-layout__nav-divider" />
926
+ } @else if (item.type === 'external') {
927
+ <a
928
+ class="ui-layout__nav-link"
929
+ [href]="item.href"
930
+ [target]="item.target || '_blank'"
931
+ [attr.rel]="item.target === '_blank' ? 'noopener noreferrer' : null"
932
+ (click)="onNavClick()"
933
+ >
934
+ @if (item.icon) {
935
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
936
+ }
937
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
938
+ <lucide-icon name="external-link" [size]="12" aria-hidden="true" class="ui-layout__nav-external" />
939
+ @if (getItemBadge(item)) {
940
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
941
+ }
942
+ </a>
943
+ } @else if (item.children?.length) {
944
+ <button
945
+ class="ui-layout__nav-link ui-layout__nav-link--parent"
946
+ [class.ui-layout__nav-link--active]="isParentActive(item)"
947
+ [class.ui-layout__nav-link--expanded]="expandedIds.has(item.id)"
948
+ [attr.aria-expanded]="expandedIds.has(item.id)"
949
+ (click)="toggleExpand(item)"
950
+ >
951
+ @if (item.icon) {
952
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
953
+ }
954
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
955
+ @if (getItemBadge(item)) {
956
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
957
+ }
958
+ <span class="ui-layout__nav-chevron" aria-hidden="true">
959
+ <lucide-icon name="chevron-right" [size]="14" />
960
+ </span>
961
+ </button>
962
+ @if (expandedIds.has(item.id)) {
963
+ <div class="ui-layout__nav-children">
964
+ @for (group of item.children; track group.label) {
965
+ <span class="ui-layout__nav-group-label">{{ group.label }}</span>
966
+ @for (child of group.items; track child.id) {
967
+ @if (!isItemHidden(child)) {
968
+ <a
969
+ class="ui-layout__nav-link ui-layout__nav-link--child"
970
+ [routerLink]="child.route"
971
+ routerLinkActive="ui-layout__nav-link--active"
972
+ (click)="onNavClick()"
973
+ >
974
+ <span class="ui-layout__nav-label">{{ child.label }}</span>
975
+ @if (getItemBadge(child)) {
976
+ <span class="ui-layout__badge">{{ getItemBadge(child) }}</span>
977
+ }
978
+ </a>
979
+ }
980
+ }
981
+ }
982
+ </div>
983
+ }
984
+ } @else {
985
+ <a
986
+ class="ui-layout__nav-link"
987
+ [routerLink]="item.route"
988
+ routerLinkActive="ui-layout__nav-link--active"
989
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
990
+ [attr.aria-current]="isRouteActive(item) ? 'page' : null"
991
+ (click)="onNavClick()"
992
+ >
993
+ @if (item.icon) {
994
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
995
+ }
996
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
997
+ @if (getItemBadge(item)) {
998
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
999
+ }
1000
+ </a>
1001
+ }
1002
+ </ng-template>
1003
+ `, isInline: true, styles: [".ui-layout-builder-host{display:block;height:100vh;height:100dvh}.ui-layout__skip-link{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-layout__skip-link:focus{position:fixed;top:var(--ui-spacing-2);left:var(--ui-spacing-2);z-index:calc(var(--ui-z-toast) + 10);width:auto;height:auto;clip:auto;padding:var(--ui-spacing-2) var(--ui-spacing-4);background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);border-radius:var(--ui-radius-md);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);text-decoration:none;overflow:visible;white-space:nowrap;margin:0}.ui-layout{display:flex;flex-direction:column;height:100%;overflow:hidden;font-family:var(--ui-font-family);color:var(--ui-color-text);background:var(--ui-color-bg-subtle)}.ui-layout__loader-overlay{position:fixed;inset:0;z-index:var(--ui-z-modal);display:flex;align-items:center;justify-content:center;background:var(--ui-color-bg-subtle);will-change:opacity;animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__loader-overlay{animation:none}}.ui-layout__loader-spinner{width:48px;height:48px}.ui-layout__loader-svg{width:100%;height:100%;animation:ui-layout-spin 1s linear infinite}.ui-layout__loader-svg circle{stroke:var(--ui-color-primary);stroke-linecap:round;stroke-dasharray:90,150;stroke-dashoffset:0;animation:ui-layout-dash 1.5s ease-in-out infinite}@media (prefers-reduced-motion: reduce){.ui-layout__loader-svg{animation:none}.ui-layout__loader-svg circle{animation:none;stroke-dasharray:90,150}}@keyframes ui-layout-spin{to{transform:rotate(360deg)}}@keyframes ui-layout-dash{0%{stroke-dasharray:1,150;stroke-dashoffset:0}50%{stroke-dasharray:90,150;stroke-dashoffset:-35}to{stroke-dasharray:90,150;stroke-dashoffset:-124}}@keyframes ui-layout-fade-in{0%{opacity:0}to{opacity:1}}.ui-layout__topbar{display:none;position:fixed;top:0;left:0;right:0;height:56px;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);align-items:center;padding:0 var(--ui-spacing-4);gap:var(--ui-spacing-3);z-index:var(--ui-z-fixed)}.ui-layout__hamburger{display:flex;flex-direction:column;justify-content:center;align-items:center;gap:5px;width:36px;height:36px;background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);cursor:pointer;padding:0;transition:all var(--ui-transition-fast);color:var(--ui-color-text)}.ui-layout__hamburger:hover{background:var(--ui-color-surface-hover)}.ui-layout__hamburger:focus{outline:none}.ui-layout__hamburger:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__hamburger-line{display:block;width:18px;height:2px;background:currentColor;border-radius:1px}.ui-layout__topbar-logo,.ui-layout__sidebar-logo{display:flex;align-items:center;justify-content:center;width:28px;height:28px;background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);border-radius:var(--ui-radius-md);font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-bold);flex-shrink:0}.ui-layout__topbar-logo-img,.ui-layout__sidebar-logo-img{width:28px;height:28px;border-radius:var(--ui-radius-md);object-fit:contain;flex-shrink:0}.ui-layout__topbar-title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__backdrop{display:none;position:fixed;inset:0;background:#0006;z-index:calc(var(--ui-z-modal-backdrop));animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__backdrop{animation:none}}.ui-layout__backdrop--topbar{display:none}.ui-layout__body{display:flex;flex:1;overflow:hidden}.ui-layout__sidebar{width:260px;min-width:260px;background:var(--ui-color-surface);border-right:1px solid var(--ui-color-border);display:flex;flex-direction:column;transition:width var(--ui-transition-normal),min-width var(--ui-transition-normal);overflow-y:auto;overflow-x:hidden}@media (prefers-reduced-motion: reduce){.ui-layout__sidebar{transition:none}}.ui-layout__sidebar--collapsed{width:60px;min-width:60px}.ui-layout__sidebar--topbar-drawer{display:none}.ui-layout__sidebar-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);flex-shrink:0}.ui-layout__sidebar--collapsed .ui-layout__sidebar-header{flex-direction:column;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3) var(--ui-spacing-2)}.ui-layout__sidebar-brand{display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-layout__sidebar-title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__collapse-btn{display:flex;align-items:center;justify-content:center;width:28px;height:28px;padding:0;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-full);cursor:pointer;color:var(--ui-color-text-muted);flex-shrink:0;transition:all var(--ui-transition-fast)}.ui-layout__collapse-btn:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text);border-color:var(--ui-color-text-muted)}.ui-layout__collapse-btn:focus{outline:none}.ui-layout__collapse-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__collapse-icon{transition:transform var(--ui-transition-normal)}.ui-layout__sidebar--collapsed .ui-layout__collapse-icon{transform:rotate(180deg)}@media (prefers-reduced-motion: reduce){.ui-layout__collapse-icon{transition:none}}.ui-layout__close-btn{display:flex;align-items:center;justify-content:center;background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);cursor:pointer;padding:var(--ui-spacing-1);color:var(--ui-color-text-muted);transition:all var(--ui-transition-fast)}.ui-layout__close-btn:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__close-btn:focus{outline:none}.ui-layout__close-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__sidebar-nav{flex:1;padding:var(--ui-spacing-3) 0;overflow-y:auto}.ui-layout__nav-section{padding:0 var(--ui-spacing-3);margin-bottom:var(--ui-spacing-4)}.ui-layout__nav-section-title{display:block;font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-muted);text-transform:uppercase;letter-spacing:.05em;padding:var(--ui-spacing-1) var(--ui-spacing-3);margin-bottom:var(--ui-spacing-1)}.ui-layout__nav-link{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);border-radius:var(--ui-radius-md);color:var(--ui-color-text-secondary);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);text-decoration:none;transition:all var(--ui-transition-fast);cursor:pointer}.ui-layout__nav-link:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text);text-decoration:none}.ui-layout__nav-link:focus{outline:none}.ui-layout__nav-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__nav-link--active{background:var(--ui-color-primary-light);color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__nav-link--parent{width:100%;background:none;border:none;text-align:left;font-family:inherit;justify-content:flex-start}.ui-layout__nav-link--child{font-size:.8rem;padding:.25rem var(--ui-spacing-2)}.ui-layout__nav-icon{flex-shrink:0}.ui-layout__nav-label{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__nav-external{flex-shrink:0;opacity:.5}.ui-layout__nav-chevron{flex-shrink:0;display:inline-flex;align-items:center;color:var(--ui-color-text-muted);transition:transform var(--ui-transition-fast)}.ui-layout__nav-link--expanded .ui-layout__nav-chevron{transform:rotate(90deg)}.ui-layout__nav-link--active .ui-layout__nav-chevron{color:var(--ui-color-primary)}@media (prefers-reduced-motion: reduce){.ui-layout__nav-chevron{transition:none}}.ui-layout__nav-children{margin-top:var(--ui-spacing-1);margin-left:calc(var(--ui-spacing-3) + 4px);padding-left:var(--ui-spacing-2);border-left:2px solid var(--ui-color-border)}.ui-layout__nav-group-label{display:block;font-size:.65rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;color:var(--ui-color-text-muted);padding:var(--ui-spacing-1) var(--ui-spacing-2);margin-top:var(--ui-spacing-2)}.ui-layout__nav-group-label:first-child{margin-top:0}.ui-layout__nav-divider{border:none;border-top:1px solid var(--ui-color-border);margin:var(--ui-spacing-2) var(--ui-spacing-3)}.ui-layout__badge{display:inline-flex;align-items:center;justify-content:center;min-width:18px;height:18px;padding:0 var(--ui-spacing-1);border-radius:var(--ui-radius-full);background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);font-size:.65rem;font-weight:var(--ui-font-weight-bold);line-height:1;flex-shrink:0}.ui-layout__sidebar-footer{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-top:1px solid var(--ui-color-border);flex-shrink:0;display:none}.ui-layout__version-badge{display:inline-block;padding:2px 8px;background:var(--ui-color-bg-muted);border-radius:var(--ui-radius-full);font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:var(--ui-font-family-mono)}.ui-layout__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-8);background:var(--ui-color-bg-subtle)}.ui-layout__content:focus{outline:none}.ui-layout__content--boxed{max-width:var(--ui-layout-content-max-width, 1200px);margin-left:auto;margin-right:auto;width:100%}.ui-layout__content--fullscreen{padding:0}.ui-layout__content--topbar{flex:1;overflow-y:auto}.ui-layout__content--transitioning{animation:ui-layout-page-enter var(--ui-transition-normal, .25s) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__content--transitioning{animation:none}}@keyframes ui-layout-page-enter{0%{opacity:0;filter:blur(4px);transform:translateY(8px)}to{opacity:1;filter:blur(0);transform:translateY(0)}}.ui-layout__page-header{margin-bottom:var(--ui-spacing-5)}.ui-layout__breadcrumb-nav{margin-bottom:var(--ui-spacing-2)}.ui-layout__breadcrumb{display:flex;flex-wrap:wrap;align-items:center;list-style:none;margin:0;padding:0;font-family:var(--ui-font-family);font-size:var(--ui-font-size-sm);line-height:var(--ui-line-height-normal);gap:var(--ui-spacing-1)}.ui-layout__breadcrumb-item{display:inline-flex;align-items:center;gap:var(--ui-spacing-1)}.ui-layout__breadcrumb-link{display:inline-flex;align-items:center;gap:4px;text-decoration:none;color:var(--ui-color-primary);border-radius:var(--ui-radius-sm);padding:2px 4px;margin:-2px -4px;transition:color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-layout__breadcrumb-link:hover{color:var(--ui-color-primary-hover);text-decoration:underline}.ui-layout__breadcrumb-link:focus{outline:none}.ui-layout__breadcrumb-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__breadcrumb-link lucide-icon{transform:translateY(2px)}.ui-layout__breadcrumb-text{white-space:nowrap}.ui-layout__breadcrumb-text--current{font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text)}.ui-layout__breadcrumb-sep{display:inline-flex;align-items:center;color:var(--ui-color-text-muted);flex-shrink:0}.ui-layout__breadcrumb-sep lucide-icon{transform:translateY(3px)}.ui-layout__page-title{margin:0;font-family:var(--ui-font-family);font-size:var(--ui-font-size-2xl);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);line-height:var(--ui-line-height-tight)}.ui-layout__page-title:focus{outline:none}.ui-layout__footer{display:flex;align-items:center;justify-content:center;gap:var(--ui-spacing-4);padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border);flex-shrink:0;flex-wrap:wrap}.ui-layout__footer-links{display:flex;gap:var(--ui-spacing-3);flex-wrap:wrap}.ui-layout__footer-link{display:inline-flex;align-items:center;gap:4px;color:var(--ui-color-text-muted);font-size:var(--ui-font-size-xs);text-decoration:none;transition:color var(--ui-transition-fast)}.ui-layout__footer-link:hover{color:var(--ui-color-primary);text-decoration:underline}.ui-layout__footer-link:focus{outline:none}.ui-layout__footer-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__footer-text{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:var(--ui-font-family-mono)}.ui-layout__fab{position:fixed;bottom:var(--ui-spacing-6);right:var(--ui-spacing-6);z-index:var(--ui-z-fixed);display:flex;flex-direction:column-reverse;align-items:center;gap:var(--ui-spacing-3)}.ui-layout__fab--bottom-left{right:auto;left:var(--ui-spacing-6)}.ui-layout__fab-main{width:56px;height:56px;border-radius:var(--ui-radius-full);border:none;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:var(--ui-shadow-lg);transition:all var(--ui-transition-fast)}.ui-layout__fab-main--primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__fab-main--primary:hover{filter:brightness(1.1)}.ui-layout__fab-main--accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__fab-main--accent:hover{filter:brightness(1.1)}.ui-layout__fab-main--warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__fab-main--warn:hover{filter:brightness(1.1)}.ui-layout__fab-main--neutral{background:var(--ui-color-surface);color:var(--ui-color-text);border:1px solid var(--ui-color-border)}.ui-layout__fab-main--neutral:hover{background:var(--ui-color-surface-hover)}.ui-layout__fab-main:focus{outline:none}.ui-layout__fab-main:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__fab-main-icon{transition:transform var(--ui-transition-fast)}.ui-layout__fab--open .ui-layout__fab-main-icon{transform:rotate(45deg)}@media (prefers-reduced-motion: reduce){.ui-layout__fab-main-icon{transition:none}}.ui-layout__fab-actions{display:flex;flex-direction:column-reverse;align-items:center;gap:var(--ui-spacing-2)}.ui-layout__fab-action{width:44px;height:44px;border-radius:var(--ui-radius-full);border:none;cursor:pointer;display:flex;align-items:center;justify-content:center;position:relative;box-shadow:var(--ui-shadow-md);animation:ui-layout-fab-pop-in var(--ui-transition-fast) ease-out both;transition:all var(--ui-transition-fast)}.ui-layout__fab-action--primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__fab-action--accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__fab-action--warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__fab-action--neutral,.ui-layout__fab-action--ghost,.ui-layout__fab-action--outline{background:var(--ui-color-surface);color:var(--ui-color-text);border:1px solid var(--ui-color-border)}.ui-layout__fab-action:hover{filter:brightness(1.1)}.ui-layout__fab-action:focus{outline:none}.ui-layout__fab-action:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}@media (prefers-reduced-motion: reduce){.ui-layout__fab-action{animation:none}}.ui-layout__fab-action-label{position:absolute;right:calc(100% + var(--ui-spacing-2));white-space:nowrap;background:var(--ui-color-neutral-900);color:#fff;font-size:var(--ui-font-size-xs);padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);pointer-events:none;box-shadow:var(--ui-shadow-sm)}.ui-layout__fab--bottom-left .ui-layout__fab-action-label{right:auto;left:calc(100% + var(--ui-spacing-2))}@keyframes ui-layout-fab-pop-in{0%{opacity:0;transform:scale(.3) translateY(10px)}to{opacity:1;transform:scale(1) translateY(0)}}.ui-layout__bottom-nav{display:none;position:fixed;bottom:0;left:0;right:0;height:56px;padding-bottom:env(safe-area-inset-bottom,0);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border);z-index:var(--ui-z-fixed);justify-content:space-around;align-items:stretch}.ui-layout__bottom-nav-item{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:2px;flex:1;color:var(--ui-color-text-muted);text-decoration:none;font-size:.65rem;font-weight:var(--ui-font-weight-medium);padding:var(--ui-spacing-1) 0;position:relative;transition:color var(--ui-transition-fast)}.ui-layout__bottom-nav-item:hover{color:var(--ui-color-text)}.ui-layout__bottom-nav-item--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__bottom-nav-item:focus{outline:none}.ui-layout__bottom-nav-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bottom-nav-label{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:64px;text-align:center}.ui-layout__desktop-only{display:initial}.ui-layout__mobile-only{display:none}.ui-layout__bar-stack{display:flex;flex-direction:column;flex-shrink:0}.ui-layout__bar{display:flex;align-items:center;padding:0 var(--ui-spacing-4);font-size:var(--ui-font-size-sm)}.ui-layout__bar--notification{min-height:44px;gap:var(--ui-spacing-2);justify-content:center;font-weight:var(--ui-font-weight-medium)}.ui-layout__bar--notification-primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__bar--notification-accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__bar--notification-warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__bar--notification-neutral{background:var(--ui-color-surface);color:var(--ui-color-text);border-bottom:1px solid var(--ui-color-border)}.ui-layout__bar--notification-ghost,.ui-layout__bar--notification-outline{background:var(--ui-color-bg-muted);color:var(--ui-color-text);border-bottom:1px solid var(--ui-color-border)}.ui-layout__bar--brand{min-height:56px;justify-content:space-between;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-4)}.ui-layout__bar--navigation{position:relative;min-height:44px;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-1);overflow-x:auto;overflow-y:hidden;flex-wrap:nowrap;scrollbar-width:none;-webkit-overflow-scrolling:touch;mask-image:linear-gradient(to right,transparent 0,black var(--ui-spacing-4),black calc(100% - var(--ui-spacing-4)),transparent 100%)}.ui-layout__bar--navigation::-webkit-scrollbar{display:none}.ui-layout__bar--links{min-height:36px;background:var(--ui-color-bg-muted);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-layout__bar-text{flex:1;text-align:center}.ui-layout__bar-dismiss{display:flex;align-items:center;justify-content:center;background:none;border:none;cursor:pointer;padding:var(--ui-spacing-1);border-radius:var(--ui-radius-sm);color:inherit;opacity:.8;transition:opacity var(--ui-transition-fast)}.ui-layout__bar-dismiss:hover{opacity:1}.ui-layout__bar-dismiss:focus{outline:none}.ui-layout__bar-dismiss:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-brand-left{display:flex;align-items:center;gap:var(--ui-spacing-3);min-width:0}.ui-layout__bar-logo,.ui-layout__bar-logo-img{width:32px;height:32px;border-radius:var(--ui-radius-md);flex-shrink:0}.ui-layout__bar-logo{display:flex;align-items:center;justify-content:center;background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-bold)}.ui-layout__bar-logo-img{object-fit:contain}.ui-layout__bar-title{font-size:var(--ui-font-size-base);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__topbar-hamburger{display:none}.ui-layout__user-dropdown{position:relative}.ui-layout__user-dropdown-trigger{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-1) var(--ui-spacing-2);background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);cursor:pointer;font-family:inherit;font-size:var(--ui-font-size-sm);color:var(--ui-color-text);transition:all var(--ui-transition-fast)}.ui-layout__user-dropdown-trigger:hover{background:var(--ui-color-surface-hover)}.ui-layout__user-dropdown-trigger:focus{outline:none}.ui-layout__user-dropdown-trigger:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__user-avatar{width:28px;height:28px;border-radius:var(--ui-radius-full);object-fit:cover}.ui-layout__user-label{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:120px}.ui-layout__user-chevron{transition:transform var(--ui-transition-fast);color:var(--ui-color-text-muted)}[aria-expanded=true] .ui-layout__user-chevron{transform:rotate(180deg)}.ui-layout__user-dropdown-menu{position:absolute;top:calc(100% + var(--ui-spacing-1));right:0;min-width:180px;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-md);padding:var(--ui-spacing-1) 0;z-index:var(--ui-z-dropdown);animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}.ui-layout__user-dropdown-item{display:flex;align-items:center;gap:var(--ui-spacing-2);width:100%;padding:var(--ui-spacing-2) var(--ui-spacing-3);background:none;border:none;cursor:pointer;font-family:inherit;font-size:var(--ui-font-size-sm);color:var(--ui-color-text);text-decoration:none;text-align:left;transition:background var(--ui-transition-fast)}.ui-layout__user-dropdown-item:hover{background:var(--ui-color-surface-hover)}.ui-layout__user-dropdown-item:focus{outline:none}.ui-layout__user-dropdown-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__user-dropdown-divider{border:none;border-top:1px solid var(--ui-color-border);margin:var(--ui-spacing-1) 0}.ui-layout__hnav-link{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);border-radius:var(--ui-radius-md);color:var(--ui-color-text-secondary);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);text-decoration:none;transition:all var(--ui-transition-fast);cursor:pointer;white-space:nowrap;background:none;border:none;font-family:inherit}.ui-layout__hnav-link:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__hnav-link:focus{outline:none}.ui-layout__hnav-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__hnav-link--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__hnav-link--parent{cursor:pointer}.ui-layout__hnav-link lucide-icon{transform:translateY(3px)}.ui-layout__hnav-chevron{transition:transform var(--ui-transition-fast)}.ui-layout__hnav-dropdown--open .ui-layout__hnav-chevron{transform:rotate(180deg)}.ui-layout__hnav-dropdown{position:relative}.ui-layout__hnav-dropdown-menu{position:absolute;top:calc(100% + var(--ui-spacing-1));left:0;min-width:200px;max-height:400px;overflow-y:auto;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-md);padding:var(--ui-spacing-2);z-index:var(--ui-z-dropdown);animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}.ui-layout__hnav-dropdown-label{display:block;font-size:.65rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;color:var(--ui-color-text-muted);padding:var(--ui-spacing-1) var(--ui-spacing-2);margin-top:var(--ui-spacing-2)}.ui-layout__hnav-dropdown-label:first-child{margin-top:0}.ui-layout__hnav-dropdown-item{display:block;padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);text-decoration:none;transition:all var(--ui-transition-fast)}.ui-layout__hnav-dropdown-item:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__hnav-dropdown-item--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__hnav-dropdown-item:focus{outline:none}.ui-layout__hnav-dropdown-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-link{display:inline-flex;align-items:center;gap:4px;color:var(--ui-color-text-muted);text-decoration:none;transition:color var(--ui-transition-fast)}.ui-layout__bar-link:hover{color:var(--ui-color-primary)}.ui-layout__bar-link:focus{outline:none}.ui-layout__bar-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-link lucide-icon{transform:translateY(3px)}@media (max-width: 767px){.ui-layout__sidebar-footer{display:block}.ui-layout__topbar{display:flex}.ui-layout:not(.ui-layout--topbar){padding-top:56px}.ui-layout__backdrop,.ui-layout__backdrop--topbar{display:block}.ui-layout__sidebar{position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar--open{transform:translate(0)}.ui-layout__sidebar--collapsed{width:300px;min-width:300px}.ui-layout__sidebar--collapsed .ui-layout__sidebar-header{flex-direction:row;align-items:center;gap:0;padding:var(--ui-spacing-4)}.ui-layout__sidebar--topbar-drawer{display:flex}}@media (max-width: 767px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar{transition:none}}@media (max-width: 767px){.ui-layout__desktop-only{display:none!important}.ui-layout__mobile-only{display:initial!important}.ui-layout__content{padding:var(--ui-spacing-4)}.ui-layout__content--fullscreen{padding:0}.ui-layout__bottom-nav{display:flex}.ui-layout__fab{bottom:calc(var(--ui-spacing-4) + 56px + env(safe-area-inset-bottom,0))}.ui-layout__topbar-hamburger{display:flex}.ui-layout__bar--navigation,.ui-layout__bar--links,.ui-layout__user-label{display:none}.ui-layout__page-title{font-size:var(--ui-font-size-xl)}.ui-layout__breadcrumb{font-size:var(--ui-font-size-xs)}}@media (min-width: 768px) and (max-width: 1023px){.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer){position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer).ui-layout__sidebar--open{transform:translate(0)}}@media (min-width: 768px) and (max-width: 1023px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer){transition:none}}@media (min-width: 768px) and (max-width: 1023px){.ui-layout__topbar{display:flex}.ui-layout:not(.ui-layout--topbar){padding-top:56px}.ui-layout__backdrop,.ui-layout__backdrop--topbar{display:block}.ui-layout__desktop-only{display:none!important}.ui-layout__mobile-only{display:initial!important}.ui-layout__content{padding:var(--ui-spacing-6)}.ui-layout__content--fullscreen{padding:0}.ui-layout__topbar-hamburger{display:flex}.ui-layout__sidebar--topbar-drawer{display:flex;position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar--topbar-drawer.ui-layout__sidebar--open{transform:translate(0)}}@media (min-width: 768px) and (max-width: 1023px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar--topbar-drawer{transition:none}}\n"], dependencies: [{ kind: "directive", type: RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
1004
+ }
1005
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiLayoutBuilderComponent, decorators: [{
1006
+ type: Component,
1007
+ args: [{ selector: 'ui-layout-builder', standalone: true, imports: [
1008
+ RouterOutlet,
1009
+ RouterLink,
1010
+ RouterLinkActive,
1011
+ NgTemplateOutlet,
1012
+ LucideAngularModule,
1013
+ ], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: { class: 'ui-layout-builder-host' }, template: `
1014
+ <!-- Skip to content (a11y) -->
1015
+ <a class="ui-layout__skip-link" href="#ui-layout-main">
1016
+ Vai al contenuto principale
1017
+ </a>
1018
+
1019
+ <div
1020
+ class="ui-layout"
1021
+ [class.ui-layout--sidebar-collapsed]="layoutMode === 'sidebar' && sidebarCollapsed"
1022
+ [class.ui-layout--topbar]="layoutMode === 'topbar'"
1023
+ >
1024
+ <!-- ─── Overlay Loader ─────────────────────────────────────── -->
1025
+ @if (loading) {
1026
+ <div
1027
+ class="ui-layout__loader-overlay"
1028
+ role="alert"
1029
+ aria-live="assertive"
1030
+ aria-label="Caricamento in corso"
1031
+ >
1032
+ <div class="ui-layout__loader-spinner" aria-hidden="true">
1033
+ <svg viewBox="0 0 50 50" class="ui-layout__loader-svg">
1034
+ <circle cx="25" cy="25" r="20" fill="none" stroke-width="4" />
1035
+ </svg>
1036
+ </div>
1037
+ </div>
1038
+ }
1039
+
1040
+ @switch (layoutMode) {
1041
+ <!-- ════════════════════════════════════════════════════════ -->
1042
+ <!-- ═══ SIDEBAR MODE ═════════════════════════════════════ -->
1043
+ <!-- ════════════════════════════════════════════════════════ -->
1044
+ @case ('sidebar') {
1045
+ <!-- Mobile Top Bar -->
1046
+ <header class="ui-layout__topbar">
1047
+ <button
1048
+ class="ui-layout__hamburger"
1049
+ (click)="layoutService.openMobileSidebar()"
1050
+ aria-label="Apri navigazione"
1051
+ >
1052
+ <span class="ui-layout__hamburger-line"></span>
1053
+ <span class="ui-layout__hamburger-line"></span>
1054
+ <span class="ui-layout__hamburger-line"></span>
1055
+ </button>
1056
+ @if (schema.header.logo) {
1057
+ @if (schema.header.logo.src) {
1058
+ <img
1059
+ class="ui-layout__topbar-logo-img"
1060
+ [src]="schema.header.logo.src"
1061
+ [alt]="schema.header.logo.alt || schema.header.title"
1062
+ />
1063
+ } @else if (schema.header.logo.text) {
1064
+ <span class="ui-layout__topbar-logo">{{ schema.header.logo.text }}</span>
1065
+ }
1066
+ }
1067
+ <span class="ui-layout__topbar-title">{{ schema.header.title }}</span>
1068
+ </header>
1069
+
1070
+ <!-- Mobile Backdrop -->
1071
+ @if (sidebarOpen) {
1072
+ <div
1073
+ class="ui-layout__backdrop"
1074
+ (click)="layoutService.closeMobileSidebar()"
1075
+ aria-hidden="true"
1076
+ ></div>
1077
+ }
1078
+
1079
+ <!-- Body (sidebar + content) -->
1080
+ <div class="ui-layout__body">
1081
+ <aside
1082
+ class="ui-layout__sidebar"
1083
+ [class.ui-layout__sidebar--collapsed]="sidebarCollapsed"
1084
+ [class.ui-layout__sidebar--open]="sidebarOpen"
1085
+ role="navigation"
1086
+ aria-label="Navigazione principale"
1087
+ >
1088
+ <div class="ui-layout__sidebar-header">
1089
+ <div class="ui-layout__sidebar-brand">
1090
+ @if (schema.header.logo) {
1091
+ @if (schema.header.logo.src) {
1092
+ <img
1093
+ class="ui-layout__sidebar-logo-img"
1094
+ [src]="schema.header.logo.src"
1095
+ [alt]="schema.header.logo.alt || schema.header.title"
1096
+ />
1097
+ } @else if (schema.header.logo.text) {
1098
+ <span class="ui-layout__sidebar-logo">{{ schema.header.logo.text }}</span>
1099
+ }
1100
+ }
1101
+ @if (!sidebarCollapsed || sidebarOpen) {
1102
+ <span class="ui-layout__sidebar-title">{{ schema.header.title }}</span>
1103
+ }
1104
+ </div>
1105
+
1106
+ @if (schema.navigation.collapsible !== false) {
1107
+ <button
1108
+ class="ui-layout__collapse-btn ui-layout__desktop-only"
1109
+ (click)="layoutService.toggleSidebar()"
1110
+ [attr.aria-label]="sidebarCollapsed ? 'Espandi sidebar' : 'Comprimi sidebar'"
1111
+ >
1112
+ <lucide-icon
1113
+ name="chevrons-left"
1114
+ [size]="16"
1115
+ class="ui-layout__collapse-icon"
1116
+ aria-hidden="true"
1117
+ />
1118
+ </button>
1119
+ }
1120
+
1121
+ <button
1122
+ class="ui-layout__close-btn ui-layout__mobile-only"
1123
+ (click)="layoutService.closeMobileSidebar()"
1124
+ aria-label="Chiudi navigazione"
1125
+ >
1126
+ <lucide-icon name="x" [size]="16" aria-hidden="true" />
1127
+ </button>
1128
+ </div>
1129
+
1130
+ @if (!sidebarCollapsed || sidebarOpen) {
1131
+ <nav class="ui-layout__sidebar-nav">
1132
+ @if (schema.navigation.sections?.length) {
1133
+ @for (section of schema.navigation.sections; track section.id) {
1134
+ <div class="ui-layout__nav-section">
1135
+ <span class="ui-layout__nav-section-title">{{ section.label }}</span>
1136
+ @for (item of getItemsForSection(section); track item.id) {
1137
+ @if (!isItemHidden(item)) {
1138
+ <ng-container
1139
+ *ngTemplateOutlet="navItemTpl; context: { $implicit: item }"
1140
+ />
1141
+ }
1142
+ }
1143
+ </div>
1144
+ }
1145
+ } @else {
1146
+ <div class="ui-layout__nav-section">
1147
+ @for (item of schema.navigation.items; track item.id) {
1148
+ @if (!isItemHidden(item)) {
1149
+ <ng-container
1150
+ *ngTemplateOutlet="navItemTpl; context: { $implicit: item }"
1151
+ />
1152
+ }
1153
+ }
1154
+ </div>
1155
+ }
1156
+ </nav>
1157
+ }
1158
+ </aside>
1159
+
1160
+ <!-- Main content area -->
1161
+ <main
1162
+ class="ui-layout__content"
1163
+ [class.ui-layout__content--boxed]="contentType === 'boxed'"
1164
+ [class.ui-layout__content--fullscreen]="contentType === 'fullscreen'"
1165
+ [class.ui-layout__content--transitioning]="contentTransitioning"
1166
+ (animationend)="contentTransitioning = false"
1167
+ id="ui-layout-main"
1168
+ tabindex="-1"
1169
+ >
1170
+ <ng-container *ngTemplateOutlet="pageHeaderTpl" />
1171
+ <router-outlet />
1172
+ </main>
1173
+ </div>
1174
+ }
1175
+
1176
+ <!-- ════════════════════════════════════════════════════════ -->
1177
+ <!-- ═══ TOPBAR MODE ══════════════════════════════════════ -->
1178
+ <!-- ════════════════════════════════════════════════════════ -->
1179
+ @case ('topbar') {
1180
+ <!-- Bar stack -->
1181
+ <div class="ui-layout__bar-stack">
1182
+ @for (bar of visibleBars; track bar.id) {
1183
+ @switch (bar.type) {
1184
+ @case ('notification') {
1185
+ @if (bar.notification) {
1186
+ <div
1187
+ class="ui-layout__bar ui-layout__bar--notification"
1188
+ [class]="'ui-layout__bar--notification-' + (bar.notification.variant || 'primary')"
1189
+ role="alert"
1190
+ >
1191
+ @if (bar.notification.icon) {
1192
+ <lucide-icon [name]="bar.notification.icon" [size]="16" aria-hidden="true" />
1193
+ }
1194
+ <span class="ui-layout__bar-text">{{ bar.notification.text }}</span>
1195
+ @if (bar.notification.dismissible !== false) {
1196
+ <button
1197
+ class="ui-layout__bar-dismiss"
1198
+ (click)="dismissBar(bar.id)"
1199
+ aria-label="Chiudi notifica"
1200
+ >
1201
+ <lucide-icon name="x" [size]="14" aria-hidden="true" />
1202
+ </button>
1203
+ }
1204
+ </div>
1205
+ }
1206
+ }
1207
+
1208
+ @case ('brand') {
1209
+ @if (bar.brand; as brand) {
1210
+ <div class="ui-layout__bar ui-layout__bar--brand">
1211
+ <div class="ui-layout__bar-brand-left">
1212
+ <button
1213
+ class="ui-layout__hamburger ui-layout__topbar-hamburger"
1214
+ (click)="layoutService.openMobileSidebar()"
1215
+ aria-label="Apri navigazione"
1216
+ >
1217
+ <span class="ui-layout__hamburger-line"></span>
1218
+ <span class="ui-layout__hamburger-line"></span>
1219
+ <span class="ui-layout__hamburger-line"></span>
1220
+ </button>
1221
+ @if (brand.logo; as logo) {
1222
+ @if (logo.src) {
1223
+ <img
1224
+ class="ui-layout__bar-logo-img"
1225
+ [src]="logo.src"
1226
+ [alt]="logo.alt || brand.title || ''"
1227
+ />
1228
+ } @else if (logo.text) {
1229
+ <span class="ui-layout__bar-logo">{{ logo.text }}</span>
1230
+ }
1231
+ }
1232
+ @if (brand.title; as brandTitle) {
1233
+ <span class="ui-layout__bar-title">{{ brandTitle }}</span>
1234
+ }
1235
+ </div>
1236
+ @if (brand.userDropdown; as dropdown) {
1237
+ <div class="ui-layout__user-dropdown">
1238
+ <button
1239
+ class="ui-layout__user-dropdown-trigger"
1240
+ (click)="userDropdownOpen = !userDropdownOpen"
1241
+ [attr.aria-expanded]="userDropdownOpen"
1242
+ aria-haspopup="true"
1243
+ >
1244
+ @if (dropdown.avatar?.src) {
1245
+ <img
1246
+ class="ui-layout__user-avatar"
1247
+ [src]="dropdown.avatar!.src"
1248
+ [alt]="dropdown.avatar!.alt || dropdown.label"
1249
+ />
1250
+ } @else {
1251
+ <lucide-icon
1252
+ [name]="dropdown.icon || 'user'"
1253
+ [size]="18"
1254
+ aria-hidden="true"
1255
+ />
1256
+ }
1257
+ <span class="ui-layout__user-label">{{ dropdown.label }}</span>
1258
+ <lucide-icon name="chevron-down" [size]="14" aria-hidden="true" class="ui-layout__user-chevron" />
1259
+ </button>
1260
+ @if (userDropdownOpen) {
1261
+ <div class="ui-layout__user-dropdown-menu" role="menu">
1262
+ @for (dItem of dropdown.items; track dItem.id) {
1263
+ @if (dItem.divider) {
1264
+ <hr class="ui-layout__user-dropdown-divider" />
1265
+ }
1266
+ @if (dItem.route) {
1267
+ <a
1268
+ class="ui-layout__user-dropdown-item"
1269
+ [routerLink]="dItem.route"
1270
+ role="menuitem"
1271
+ (click)="userDropdownOpen = false"
1272
+ >
1273
+ @if (dItem.icon) {
1274
+ <lucide-icon [name]="dItem.icon" [size]="16" aria-hidden="true" />
1275
+ }
1276
+ {{ dItem.label }}
1277
+ </a>
1278
+ } @else {
1279
+ <button
1280
+ class="ui-layout__user-dropdown-item"
1281
+ role="menuitem"
1282
+ (click)="onDropdownAction(dItem)"
1283
+ >
1284
+ @if (dItem.icon) {
1285
+ <lucide-icon [name]="dItem.icon" [size]="16" aria-hidden="true" />
1286
+ }
1287
+ {{ dItem.label }}
1288
+ </button>
1289
+ }
1290
+ }
1291
+ </div>
1292
+ }
1293
+ </div>
1294
+ }
1295
+ </div>
1296
+ }
1297
+ }
1298
+
1299
+ @case ('navigation') {
1300
+ <nav class="ui-layout__bar ui-layout__bar--navigation" role="navigation" aria-label="Navigazione principale">
1301
+ @for (item of getTopbarNavItems(bar); track item.id) {
1302
+ @if (!isItemHidden(item)) {
1303
+ @if (item.children?.length) {
1304
+ <div class="ui-layout__hnav-dropdown" [class.ui-layout__hnav-dropdown--open]="hnavDropdownId === item.id">
1305
+ <button
1306
+ class="ui-layout__hnav-link ui-layout__hnav-link--parent"
1307
+ [class.ui-layout__hnav-link--active]="isParentActive(item)"
1308
+ (click)="toggleHnavDropdown(item.id)"
1309
+ [attr.aria-expanded]="hnavDropdownId === item.id"
1310
+ >
1311
+ @if (item.icon) {
1312
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
1313
+ }
1314
+ <span>{{ item.label }}</span>
1315
+ <lucide-icon name="chevron-down" [size]="14" aria-hidden="true" class="ui-layout__hnav-chevron" />
1316
+ </button>
1317
+ @if (hnavDropdownId === item.id) {
1318
+ <div class="ui-layout__hnav-dropdown-menu">
1319
+ @for (group of item.children; track group.label) {
1320
+ <span class="ui-layout__hnav-dropdown-label">{{ group.label }}</span>
1321
+ @for (child of group.items; track child.id) {
1322
+ @if (!isItemHidden(child)) {
1323
+ <a
1324
+ class="ui-layout__hnav-dropdown-item"
1325
+ [routerLink]="child.route"
1326
+ routerLinkActive="ui-layout__hnav-dropdown-item--active"
1327
+ (click)="hnavDropdownId = null; onNavClick()"
1328
+ >
1329
+ {{ child.label }}
1330
+ </a>
1331
+ }
1332
+ }
1333
+ }
1334
+ </div>
1335
+ }
1336
+ </div>
1337
+ } @else if (item.type === 'external') {
1338
+ <a
1339
+ class="ui-layout__hnav-link"
1340
+ [href]="item.href"
1341
+ [target]="item.target || '_blank'"
1342
+ [attr.rel]="item.target === '_blank' ? 'noopener noreferrer' : null"
1343
+ >
1344
+ @if (item.icon) {
1345
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
1346
+ }
1347
+ <span>{{ item.label }}</span>
1348
+ <lucide-icon name="external-link" [size]="12" aria-hidden="true" />
1349
+ </a>
1350
+ } @else if (item.type !== 'divider') {
1351
+ <a
1352
+ class="ui-layout__hnav-link"
1353
+ [routerLink]="item.route"
1354
+ routerLinkActive="ui-layout__hnav-link--active"
1355
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
1356
+ >
1357
+ @if (item.icon) {
1358
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" />
1359
+ }
1360
+ <span>{{ item.label }}</span>
1361
+ @if (getItemBadge(item)) {
1362
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
1363
+ }
1364
+ </a>
1365
+ }
1366
+ }
1367
+ }
1368
+ </nav>
1369
+ }
1370
+
1371
+ @case ('links') {
1372
+ @if (bar.links) {
1373
+ <div
1374
+ class="ui-layout__bar ui-layout__bar--links"
1375
+ [style.justify-content]="bar.links.align === 'start' ? 'flex-start' : bar.links.align === 'center' ? 'center' : bar.links.align === 'space-between' ? 'space-between' : 'flex-end'"
1376
+ >
1377
+ @for (lnk of bar.links.items; track lnk.label || lnk.ariaLabel || $index) {
1378
+ @if (lnk.route) {
1379
+ <a class="ui-layout__bar-link" [routerLink]="lnk.route" [attr.aria-label]="lnk.ariaLabel || null">
1380
+ @if (lnk.icon) {
1381
+ <lucide-icon [name]="lnk.icon" [size]="14" aria-hidden="true" />
1382
+ }
1383
+ @if (lnk.label) {
1384
+ <span>{{ lnk.label }}</span>
1385
+ }
1386
+ </a>
1387
+ } @else if (lnk.href) {
1388
+ <a
1389
+ class="ui-layout__bar-link"
1390
+ [href]="lnk.href"
1391
+ [target]="lnk.target || '_self'"
1392
+ [attr.rel]="lnk.target === '_blank' ? 'noopener noreferrer' : null"
1393
+ [attr.aria-label]="lnk.ariaLabel || null"
1394
+ >
1395
+ @if (lnk.icon) {
1396
+ <lucide-icon [name]="lnk.icon" [size]="14" aria-hidden="true" />
1397
+ }
1398
+ @if (lnk.label) {
1399
+ <span>{{ lnk.label }}</span>
1400
+ }
1401
+ </a>
1402
+ }
1403
+ }
1404
+ </div>
1405
+ }
1406
+ }
1407
+ }
1408
+ }
1409
+ </div>
1410
+
1411
+ <!-- Mobile Backdrop (topbar mode) -->
1412
+ @if (sidebarOpen) {
1413
+ <div
1414
+ class="ui-layout__backdrop ui-layout__backdrop--topbar"
1415
+ (click)="layoutService.closeMobileSidebar()"
1416
+ aria-hidden="true"
1417
+ ></div>
1418
+ }
1419
+
1420
+ <!-- Mobile drawer (topbar mode) -->
1421
+ <aside
1422
+ class="ui-layout__sidebar ui-layout__sidebar--topbar-drawer"
1423
+ [class.ui-layout__sidebar--open]="sidebarOpen"
1424
+ role="navigation"
1425
+ aria-label="Navigazione principale"
1426
+ >
1427
+ <div class="ui-layout__sidebar-header">
1428
+ <div class="ui-layout__sidebar-brand">
1429
+ @if (schema.header.logo) {
1430
+ @if (schema.header.logo.src) {
1431
+ <img class="ui-layout__sidebar-logo-img" [src]="schema.header.logo.src" [alt]="schema.header.logo.alt || schema.header.title" />
1432
+ } @else if (schema.header.logo.text) {
1433
+ <span class="ui-layout__sidebar-logo">{{ schema.header.logo.text }}</span>
1434
+ }
1435
+ }
1436
+ <span class="ui-layout__sidebar-title">{{ schema.header.title }}</span>
1437
+ </div>
1438
+ <button class="ui-layout__close-btn" (click)="layoutService.closeMobileSidebar()" aria-label="Chiudi navigazione">
1439
+ <lucide-icon name="x" [size]="16" aria-hidden="true" />
1440
+ </button>
1441
+ </div>
1442
+ <nav class="ui-layout__sidebar-nav">
1443
+ @if (schema.navigation.sections?.length) {
1444
+ @for (section of schema.navigation.sections; track section.id) {
1445
+ <div class="ui-layout__nav-section">
1446
+ <span class="ui-layout__nav-section-title">{{ section.label }}</span>
1447
+ @for (item of getItemsForSection(section); track item.id) {
1448
+ @if (!isItemHidden(item)) {
1449
+ <ng-container *ngTemplateOutlet="navItemTpl; context: { $implicit: item }" />
1450
+ }
1451
+ }
1452
+ </div>
1453
+ }
1454
+ } @else {
1455
+ <div class="ui-layout__nav-section">
1456
+ @for (item of schema.navigation.items; track item.id) {
1457
+ @if (!isItemHidden(item)) {
1458
+ <ng-container *ngTemplateOutlet="navItemTpl; context: { $implicit: item }" />
1459
+ }
1460
+ }
1461
+ </div>
1462
+ }
1463
+ </nav>
1464
+ </aside>
1465
+
1466
+ <!-- Main content area (topbar mode) -->
1467
+ <main
1468
+ class="ui-layout__content ui-layout__content--topbar"
1469
+ [class.ui-layout__content--boxed]="contentType === 'boxed'"
1470
+ [class.ui-layout__content--fullscreen]="contentType === 'fullscreen'"
1471
+ [class.ui-layout__content--transitioning]="contentTransitioning"
1472
+ (animationend)="contentTransitioning = false"
1473
+ id="ui-layout-main"
1474
+ tabindex="-1"
1475
+ >
1476
+ <ng-container *ngTemplateOutlet="pageHeaderTpl" />
1477
+ <router-outlet />
1478
+ </main>
1479
+ }
1480
+ }
1481
+
1482
+ <!-- ─── Footer ─────────────────────────────────────────────── -->
1483
+ @if (schema.footer) {
1484
+ <footer class="ui-layout__footer">
1485
+ @if (schema.footer.links?.length) {
1486
+ <nav class="ui-layout__footer-links" aria-label="Link footer">
1487
+ @for (link of schema.footer.links; track link.label) {
1488
+ @if (link.route) {
1489
+ <a class="ui-layout__footer-link" [routerLink]="link.route">
1490
+ @if (link.icon) {
1491
+ <lucide-icon [name]="link.icon" [size]="14" aria-hidden="true" />
1492
+ }
1493
+ {{ link.label }}
1494
+ </a>
1495
+ } @else if (link.href) {
1496
+ <a
1497
+ class="ui-layout__footer-link"
1498
+ [href]="link.href"
1499
+ [target]="link.target || '_self'"
1500
+ [attr.rel]="link.target === '_blank' ? 'noopener noreferrer' : null"
1501
+ >
1502
+ @if (link.icon) {
1503
+ <lucide-icon [name]="link.icon" [size]="14" aria-hidden="true" />
1504
+ }
1505
+ {{ link.label }}
1506
+ </a>
1507
+ }
1508
+ }
1509
+ </nav>
1510
+ }
1511
+ @if (schema.footer.text) {
1512
+ <span class="ui-layout__footer-text">{{ schema.footer.text }}</span>
1513
+ }
1514
+ </footer>
1515
+ }
1516
+
1517
+ <!-- ─── Speed Dial FAB ─────────────────────────────────────── -->
1518
+ @if (currentFabConfig) {
1519
+ <div
1520
+ class="ui-layout__fab"
1521
+ [class.ui-layout__fab--open]="fabOpen"
1522
+ [class.ui-layout__fab--bottom-left]="currentFabConfig.position === 'bottom-left'"
1523
+ role="complementary"
1524
+ aria-label="Azioni rapide"
1525
+ >
1526
+ @if (fabOpen && currentFabConfig.secondaryActions?.length) {
1527
+ <div class="ui-layout__fab-actions">
1528
+ @for (action of currentFabConfig.secondaryActions; track action.id; let i = $index) {
1529
+ <button
1530
+ class="ui-layout__fab-action"
1531
+ [class]="'ui-layout__fab-action--' + (action.variant || 'neutral')"
1532
+ [attr.aria-label]="action.ariaLabel || action.label"
1533
+ [title]="action.label"
1534
+ [style.animation-delay]="(i * 50) + 'ms'"
1535
+ (click)="onFabAction(action)"
1536
+ >
1537
+ <lucide-icon [name]="action.icon" [size]="18" aria-hidden="true" />
1538
+ <span class="ui-layout__fab-action-label">{{ action.label }}</span>
1539
+ </button>
1540
+ }
1541
+ </div>
1542
+ }
1543
+ <button
1544
+ class="ui-layout__fab-main"
1545
+ [class]="'ui-layout__fab-main--' + (currentFabConfig.mainAction.variant || 'primary')"
1546
+ [attr.aria-label]="currentFabConfig.mainAction.ariaLabel || currentFabConfig.mainAction.label"
1547
+ [attr.aria-expanded]="currentFabConfig.secondaryActions?.length ? fabOpen : null"
1548
+ (click)="onFabMainClick()"
1549
+ >
1550
+ <lucide-icon
1551
+ [name]="currentFabConfig.mainAction.icon"
1552
+ [size]="24"
1553
+ aria-hidden="true"
1554
+ class="ui-layout__fab-main-icon"
1555
+ />
1556
+ </button>
1557
+ </div>
1558
+ }
1559
+
1560
+ <!-- ─── Bottom Navigation (mobile, sidebar mode) ────────────── -->
1561
+ @if (layoutMode === 'sidebar' && bottomNavItems.length > 0 && mobileMode !== 'drawer') {
1562
+ <nav
1563
+ class="ui-layout__bottom-nav"
1564
+ role="navigation"
1565
+ aria-label="Navigazione rapida"
1566
+ >
1567
+ @for (item of bottomNavItems; track item.id) {
1568
+ <a
1569
+ class="ui-layout__bottom-nav-item"
1570
+ [routerLink]="item.route"
1571
+ routerLinkActive="ui-layout__bottom-nav-item--active"
1572
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
1573
+ [attr.aria-current]="isRouteActive(item) ? 'page' : null"
1574
+ >
1575
+ @if (item.icon) {
1576
+ <lucide-icon [name]="item.icon" [size]="20" aria-hidden="true" />
1577
+ }
1578
+ <span class="ui-layout__bottom-nav-label">{{ item.label }}</span>
1579
+ @if (getItemBadge(item)) {
1580
+ <span class="ui-layout__badge" aria-label="Notifica">{{ getItemBadge(item) }}</span>
1581
+ }
1582
+ </a>
1583
+ }
1584
+ </nav>
1585
+ }
1586
+ </div>
1587
+
1588
+ <!-- ═══ SHARED TEMPLATES ═══════════════════════════════════════ -->
1589
+
1590
+ <!-- Page header with auto-derived breadcrumbs -->
1591
+ <ng-template #pageHeaderTpl>
1592
+ @if (schema.pageHeader?.show !== false && breadcrumbs.length > 0) {
1593
+ <header class="ui-layout__page-header">
1594
+ <nav aria-label="Breadcrumb" class="ui-layout__breadcrumb-nav">
1595
+ <ol class="ui-layout__breadcrumb" itemscope itemtype="https://schema.org/BreadcrumbList">
1596
+ @if (schema.pageHeader?.showHome !== false) {
1597
+ <li class="ui-layout__breadcrumb-item" itemprop="itemListElement" itemscope itemtype="https://schema.org/ListItem">
1598
+ <a itemprop="item" [routerLink]="schema.pageHeader?.homeRoute || '/'" class="ui-layout__breadcrumb-link" aria-label="Home">
1599
+ <lucide-icon name="home" [size]="14" aria-hidden="true" />
1600
+ <span itemprop="name" class="ui-layout__breadcrumb-text">Home</span>
1601
+ </a>
1602
+ <meta itemprop="position" content="1" />
1603
+ <span class="ui-layout__breadcrumb-sep" aria-hidden="true">
1604
+ <lucide-icon name="chevron-right" [size]="14" />
1605
+ </span>
1606
+ </li>
1607
+ }
1608
+ @for (crumb of breadcrumbs; track crumb.label; let i = $index) {
1609
+ <li
1610
+ class="ui-layout__breadcrumb-item"
1611
+ [class.ui-layout__breadcrumb-item--current]="crumb.isLast"
1612
+ itemprop="itemListElement"
1613
+ itemscope
1614
+ itemtype="https://schema.org/ListItem"
1615
+ >
1616
+ @if (!crumb.isLast && crumb.url) {
1617
+ <a itemprop="item" [routerLink]="crumb.url" class="ui-layout__breadcrumb-link">
1618
+ <span itemprop="name" class="ui-layout__breadcrumb-text">{{ crumb.label }}</span>
1619
+ </a>
1620
+ <span class="ui-layout__breadcrumb-sep" aria-hidden="true">
1621
+ <lucide-icon name="chevron-right" [size]="14" />
1622
+ </span>
1623
+ } @else {
1624
+ <span itemprop="name" class="ui-layout__breadcrumb-text ui-layout__breadcrumb-text--current" aria-current="page">
1625
+ {{ crumb.label }}
1626
+ </span>
1627
+ }
1628
+ <meta itemprop="position" [attr.content]="(schema.pageHeader?.showHome !== false) ? i + 2 : i + 1" />
1629
+ </li>
1630
+ }
1631
+ </ol>
1632
+ </nav>
1633
+ <h1 class="ui-layout__page-title">{{ pageTitle }}</h1>
1634
+ </header>
1635
+ }
1636
+ </ng-template>
1637
+
1638
+ <!-- Nav item template (sidebar, recursive-friendly) -->
1639
+ <ng-template #navItemTpl let-item>
1640
+ @if (item.type === 'divider') {
1641
+ <hr class="ui-layout__nav-divider" />
1642
+ } @else if (item.type === 'external') {
1643
+ <a
1644
+ class="ui-layout__nav-link"
1645
+ [href]="item.href"
1646
+ [target]="item.target || '_blank'"
1647
+ [attr.rel]="item.target === '_blank' ? 'noopener noreferrer' : null"
1648
+ (click)="onNavClick()"
1649
+ >
1650
+ @if (item.icon) {
1651
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
1652
+ }
1653
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
1654
+ <lucide-icon name="external-link" [size]="12" aria-hidden="true" class="ui-layout__nav-external" />
1655
+ @if (getItemBadge(item)) {
1656
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
1657
+ }
1658
+ </a>
1659
+ } @else if (item.children?.length) {
1660
+ <button
1661
+ class="ui-layout__nav-link ui-layout__nav-link--parent"
1662
+ [class.ui-layout__nav-link--active]="isParentActive(item)"
1663
+ [class.ui-layout__nav-link--expanded]="expandedIds.has(item.id)"
1664
+ [attr.aria-expanded]="expandedIds.has(item.id)"
1665
+ (click)="toggleExpand(item)"
1666
+ >
1667
+ @if (item.icon) {
1668
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
1669
+ }
1670
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
1671
+ @if (getItemBadge(item)) {
1672
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
1673
+ }
1674
+ <span class="ui-layout__nav-chevron" aria-hidden="true">
1675
+ <lucide-icon name="chevron-right" [size]="14" />
1676
+ </span>
1677
+ </button>
1678
+ @if (expandedIds.has(item.id)) {
1679
+ <div class="ui-layout__nav-children">
1680
+ @for (group of item.children; track group.label) {
1681
+ <span class="ui-layout__nav-group-label">{{ group.label }}</span>
1682
+ @for (child of group.items; track child.id) {
1683
+ @if (!isItemHidden(child)) {
1684
+ <a
1685
+ class="ui-layout__nav-link ui-layout__nav-link--child"
1686
+ [routerLink]="child.route"
1687
+ routerLinkActive="ui-layout__nav-link--active"
1688
+ (click)="onNavClick()"
1689
+ >
1690
+ <span class="ui-layout__nav-label">{{ child.label }}</span>
1691
+ @if (getItemBadge(child)) {
1692
+ <span class="ui-layout__badge">{{ getItemBadge(child) }}</span>
1693
+ }
1694
+ </a>
1695
+ }
1696
+ }
1697
+ }
1698
+ </div>
1699
+ }
1700
+ } @else {
1701
+ <a
1702
+ class="ui-layout__nav-link"
1703
+ [routerLink]="item.route"
1704
+ routerLinkActive="ui-layout__nav-link--active"
1705
+ [routerLinkActiveOptions]="{ exact: item.routeActiveExact || false }"
1706
+ [attr.aria-current]="isRouteActive(item) ? 'page' : null"
1707
+ (click)="onNavClick()"
1708
+ >
1709
+ @if (item.icon) {
1710
+ <lucide-icon [name]="item.icon" [size]="16" aria-hidden="true" class="ui-layout__nav-icon" />
1711
+ }
1712
+ <span class="ui-layout__nav-label">{{ item.label }}</span>
1713
+ @if (getItemBadge(item)) {
1714
+ <span class="ui-layout__badge">{{ getItemBadge(item) }}</span>
1715
+ }
1716
+ </a>
1717
+ }
1718
+ </ng-template>
1719
+ `, styles: [".ui-layout-builder-host{display:block;height:100vh;height:100dvh}.ui-layout__skip-link{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-layout__skip-link:focus{position:fixed;top:var(--ui-spacing-2);left:var(--ui-spacing-2);z-index:calc(var(--ui-z-toast) + 10);width:auto;height:auto;clip:auto;padding:var(--ui-spacing-2) var(--ui-spacing-4);background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);border-radius:var(--ui-radius-md);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);text-decoration:none;overflow:visible;white-space:nowrap;margin:0}.ui-layout{display:flex;flex-direction:column;height:100%;overflow:hidden;font-family:var(--ui-font-family);color:var(--ui-color-text);background:var(--ui-color-bg-subtle)}.ui-layout__loader-overlay{position:fixed;inset:0;z-index:var(--ui-z-modal);display:flex;align-items:center;justify-content:center;background:var(--ui-color-bg-subtle);will-change:opacity;animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__loader-overlay{animation:none}}.ui-layout__loader-spinner{width:48px;height:48px}.ui-layout__loader-svg{width:100%;height:100%;animation:ui-layout-spin 1s linear infinite}.ui-layout__loader-svg circle{stroke:var(--ui-color-primary);stroke-linecap:round;stroke-dasharray:90,150;stroke-dashoffset:0;animation:ui-layout-dash 1.5s ease-in-out infinite}@media (prefers-reduced-motion: reduce){.ui-layout__loader-svg{animation:none}.ui-layout__loader-svg circle{animation:none;stroke-dasharray:90,150}}@keyframes ui-layout-spin{to{transform:rotate(360deg)}}@keyframes ui-layout-dash{0%{stroke-dasharray:1,150;stroke-dashoffset:0}50%{stroke-dasharray:90,150;stroke-dashoffset:-35}to{stroke-dasharray:90,150;stroke-dashoffset:-124}}@keyframes ui-layout-fade-in{0%{opacity:0}to{opacity:1}}.ui-layout__topbar{display:none;position:fixed;top:0;left:0;right:0;height:56px;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);align-items:center;padding:0 var(--ui-spacing-4);gap:var(--ui-spacing-3);z-index:var(--ui-z-fixed)}.ui-layout__hamburger{display:flex;flex-direction:column;justify-content:center;align-items:center;gap:5px;width:36px;height:36px;background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);cursor:pointer;padding:0;transition:all var(--ui-transition-fast);color:var(--ui-color-text)}.ui-layout__hamburger:hover{background:var(--ui-color-surface-hover)}.ui-layout__hamburger:focus{outline:none}.ui-layout__hamburger:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__hamburger-line{display:block;width:18px;height:2px;background:currentColor;border-radius:1px}.ui-layout__topbar-logo,.ui-layout__sidebar-logo{display:flex;align-items:center;justify-content:center;width:28px;height:28px;background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);border-radius:var(--ui-radius-md);font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-bold);flex-shrink:0}.ui-layout__topbar-logo-img,.ui-layout__sidebar-logo-img{width:28px;height:28px;border-radius:var(--ui-radius-md);object-fit:contain;flex-shrink:0}.ui-layout__topbar-title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__backdrop{display:none;position:fixed;inset:0;background:#0006;z-index:calc(var(--ui-z-modal-backdrop));animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__backdrop{animation:none}}.ui-layout__backdrop--topbar{display:none}.ui-layout__body{display:flex;flex:1;overflow:hidden}.ui-layout__sidebar{width:260px;min-width:260px;background:var(--ui-color-surface);border-right:1px solid var(--ui-color-border);display:flex;flex-direction:column;transition:width var(--ui-transition-normal),min-width var(--ui-transition-normal);overflow-y:auto;overflow-x:hidden}@media (prefers-reduced-motion: reduce){.ui-layout__sidebar{transition:none}}.ui-layout__sidebar--collapsed{width:60px;min-width:60px}.ui-layout__sidebar--topbar-drawer{display:none}.ui-layout__sidebar-header{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);flex-shrink:0}.ui-layout__sidebar--collapsed .ui-layout__sidebar-header{flex-direction:column;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-3) var(--ui-spacing-2)}.ui-layout__sidebar-brand{display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-layout__sidebar-title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__collapse-btn{display:flex;align-items:center;justify-content:center;width:28px;height:28px;padding:0;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-full);cursor:pointer;color:var(--ui-color-text-muted);flex-shrink:0;transition:all var(--ui-transition-fast)}.ui-layout__collapse-btn:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text);border-color:var(--ui-color-text-muted)}.ui-layout__collapse-btn:focus{outline:none}.ui-layout__collapse-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__collapse-icon{transition:transform var(--ui-transition-normal)}.ui-layout__sidebar--collapsed .ui-layout__collapse-icon{transform:rotate(180deg)}@media (prefers-reduced-motion: reduce){.ui-layout__collapse-icon{transition:none}}.ui-layout__close-btn{display:flex;align-items:center;justify-content:center;background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-sm);cursor:pointer;padding:var(--ui-spacing-1);color:var(--ui-color-text-muted);transition:all var(--ui-transition-fast)}.ui-layout__close-btn:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__close-btn:focus{outline:none}.ui-layout__close-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__sidebar-nav{flex:1;padding:var(--ui-spacing-3) 0;overflow-y:auto}.ui-layout__nav-section{padding:0 var(--ui-spacing-3);margin-bottom:var(--ui-spacing-4)}.ui-layout__nav-section-title{display:block;font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-muted);text-transform:uppercase;letter-spacing:.05em;padding:var(--ui-spacing-1) var(--ui-spacing-3);margin-bottom:var(--ui-spacing-1)}.ui-layout__nav-link{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);border-radius:var(--ui-radius-md);color:var(--ui-color-text-secondary);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);text-decoration:none;transition:all var(--ui-transition-fast);cursor:pointer}.ui-layout__nav-link:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text);text-decoration:none}.ui-layout__nav-link:focus{outline:none}.ui-layout__nav-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__nav-link--active{background:var(--ui-color-primary-light);color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__nav-link--parent{width:100%;background:none;border:none;text-align:left;font-family:inherit;justify-content:flex-start}.ui-layout__nav-link--child{font-size:.8rem;padding:.25rem var(--ui-spacing-2)}.ui-layout__nav-icon{flex-shrink:0}.ui-layout__nav-label{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__nav-external{flex-shrink:0;opacity:.5}.ui-layout__nav-chevron{flex-shrink:0;display:inline-flex;align-items:center;color:var(--ui-color-text-muted);transition:transform var(--ui-transition-fast)}.ui-layout__nav-link--expanded .ui-layout__nav-chevron{transform:rotate(90deg)}.ui-layout__nav-link--active .ui-layout__nav-chevron{color:var(--ui-color-primary)}@media (prefers-reduced-motion: reduce){.ui-layout__nav-chevron{transition:none}}.ui-layout__nav-children{margin-top:var(--ui-spacing-1);margin-left:calc(var(--ui-spacing-3) + 4px);padding-left:var(--ui-spacing-2);border-left:2px solid var(--ui-color-border)}.ui-layout__nav-group-label{display:block;font-size:.65rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;color:var(--ui-color-text-muted);padding:var(--ui-spacing-1) var(--ui-spacing-2);margin-top:var(--ui-spacing-2)}.ui-layout__nav-group-label:first-child{margin-top:0}.ui-layout__nav-divider{border:none;border-top:1px solid var(--ui-color-border);margin:var(--ui-spacing-2) var(--ui-spacing-3)}.ui-layout__badge{display:inline-flex;align-items:center;justify-content:center;min-width:18px;height:18px;padding:0 var(--ui-spacing-1);border-radius:var(--ui-radius-full);background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);font-size:.65rem;font-weight:var(--ui-font-weight-bold);line-height:1;flex-shrink:0}.ui-layout__sidebar-footer{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-top:1px solid var(--ui-color-border);flex-shrink:0;display:none}.ui-layout__version-badge{display:inline-block;padding:2px 8px;background:var(--ui-color-bg-muted);border-radius:var(--ui-radius-full);font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:var(--ui-font-family-mono)}.ui-layout__content{flex:1;overflow-y:auto;padding:var(--ui-spacing-8);background:var(--ui-color-bg-subtle)}.ui-layout__content:focus{outline:none}.ui-layout__content--boxed{max-width:var(--ui-layout-content-max-width, 1200px);margin-left:auto;margin-right:auto;width:100%}.ui-layout__content--fullscreen{padding:0}.ui-layout__content--topbar{flex:1;overflow-y:auto}.ui-layout__content--transitioning{animation:ui-layout-page-enter var(--ui-transition-normal, .25s) ease-out}@media (prefers-reduced-motion: reduce){.ui-layout__content--transitioning{animation:none}}@keyframes ui-layout-page-enter{0%{opacity:0;filter:blur(4px);transform:translateY(8px)}to{opacity:1;filter:blur(0);transform:translateY(0)}}.ui-layout__page-header{margin-bottom:var(--ui-spacing-5)}.ui-layout__breadcrumb-nav{margin-bottom:var(--ui-spacing-2)}.ui-layout__breadcrumb{display:flex;flex-wrap:wrap;align-items:center;list-style:none;margin:0;padding:0;font-family:var(--ui-font-family);font-size:var(--ui-font-size-sm);line-height:var(--ui-line-height-normal);gap:var(--ui-spacing-1)}.ui-layout__breadcrumb-item{display:inline-flex;align-items:center;gap:var(--ui-spacing-1)}.ui-layout__breadcrumb-link{display:inline-flex;align-items:center;gap:4px;text-decoration:none;color:var(--ui-color-primary);border-radius:var(--ui-radius-sm);padding:2px 4px;margin:-2px -4px;transition:color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-layout__breadcrumb-link:hover{color:var(--ui-color-primary-hover);text-decoration:underline}.ui-layout__breadcrumb-link:focus{outline:none}.ui-layout__breadcrumb-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__breadcrumb-link lucide-icon{transform:translateY(2px)}.ui-layout__breadcrumb-text{white-space:nowrap}.ui-layout__breadcrumb-text--current{font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text)}.ui-layout__breadcrumb-sep{display:inline-flex;align-items:center;color:var(--ui-color-text-muted);flex-shrink:0}.ui-layout__breadcrumb-sep lucide-icon{transform:translateY(3px)}.ui-layout__page-title{margin:0;font-family:var(--ui-font-family);font-size:var(--ui-font-size-2xl);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);line-height:var(--ui-line-height-tight)}.ui-layout__page-title:focus{outline:none}.ui-layout__footer{display:flex;align-items:center;justify-content:center;gap:var(--ui-spacing-4);padding:var(--ui-spacing-3) var(--ui-spacing-4);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border);flex-shrink:0;flex-wrap:wrap}.ui-layout__footer-links{display:flex;gap:var(--ui-spacing-3);flex-wrap:wrap}.ui-layout__footer-link{display:inline-flex;align-items:center;gap:4px;color:var(--ui-color-text-muted);font-size:var(--ui-font-size-xs);text-decoration:none;transition:color var(--ui-transition-fast)}.ui-layout__footer-link:hover{color:var(--ui-color-primary);text-decoration:underline}.ui-layout__footer-link:focus{outline:none}.ui-layout__footer-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__footer-text{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);font-family:var(--ui-font-family-mono)}.ui-layout__fab{position:fixed;bottom:var(--ui-spacing-6);right:var(--ui-spacing-6);z-index:var(--ui-z-fixed);display:flex;flex-direction:column-reverse;align-items:center;gap:var(--ui-spacing-3)}.ui-layout__fab--bottom-left{right:auto;left:var(--ui-spacing-6)}.ui-layout__fab-main{width:56px;height:56px;border-radius:var(--ui-radius-full);border:none;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:var(--ui-shadow-lg);transition:all var(--ui-transition-fast)}.ui-layout__fab-main--primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__fab-main--primary:hover{filter:brightness(1.1)}.ui-layout__fab-main--accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__fab-main--accent:hover{filter:brightness(1.1)}.ui-layout__fab-main--warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__fab-main--warn:hover{filter:brightness(1.1)}.ui-layout__fab-main--neutral{background:var(--ui-color-surface);color:var(--ui-color-text);border:1px solid var(--ui-color-border)}.ui-layout__fab-main--neutral:hover{background:var(--ui-color-surface-hover)}.ui-layout__fab-main:focus{outline:none}.ui-layout__fab-main:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__fab-main-icon{transition:transform var(--ui-transition-fast)}.ui-layout__fab--open .ui-layout__fab-main-icon{transform:rotate(45deg)}@media (prefers-reduced-motion: reduce){.ui-layout__fab-main-icon{transition:none}}.ui-layout__fab-actions{display:flex;flex-direction:column-reverse;align-items:center;gap:var(--ui-spacing-2)}.ui-layout__fab-action{width:44px;height:44px;border-radius:var(--ui-radius-full);border:none;cursor:pointer;display:flex;align-items:center;justify-content:center;position:relative;box-shadow:var(--ui-shadow-md);animation:ui-layout-fab-pop-in var(--ui-transition-fast) ease-out both;transition:all var(--ui-transition-fast)}.ui-layout__fab-action--primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__fab-action--accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__fab-action--warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__fab-action--neutral,.ui-layout__fab-action--ghost,.ui-layout__fab-action--outline{background:var(--ui-color-surface);color:var(--ui-color-text);border:1px solid var(--ui-color-border)}.ui-layout__fab-action:hover{filter:brightness(1.1)}.ui-layout__fab-action:focus{outline:none}.ui-layout__fab-action:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}@media (prefers-reduced-motion: reduce){.ui-layout__fab-action{animation:none}}.ui-layout__fab-action-label{position:absolute;right:calc(100% + var(--ui-spacing-2));white-space:nowrap;background:var(--ui-color-neutral-900);color:#fff;font-size:var(--ui-font-size-xs);padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);pointer-events:none;box-shadow:var(--ui-shadow-sm)}.ui-layout__fab--bottom-left .ui-layout__fab-action-label{right:auto;left:calc(100% + var(--ui-spacing-2))}@keyframes ui-layout-fab-pop-in{0%{opacity:0;transform:scale(.3) translateY(10px)}to{opacity:1;transform:scale(1) translateY(0)}}.ui-layout__bottom-nav{display:none;position:fixed;bottom:0;left:0;right:0;height:56px;padding-bottom:env(safe-area-inset-bottom,0);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border);z-index:var(--ui-z-fixed);justify-content:space-around;align-items:stretch}.ui-layout__bottom-nav-item{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:2px;flex:1;color:var(--ui-color-text-muted);text-decoration:none;font-size:.65rem;font-weight:var(--ui-font-weight-medium);padding:var(--ui-spacing-1) 0;position:relative;transition:color var(--ui-transition-fast)}.ui-layout__bottom-nav-item:hover{color:var(--ui-color-text)}.ui-layout__bottom-nav-item--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__bottom-nav-item:focus{outline:none}.ui-layout__bottom-nav-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bottom-nav-label{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:64px;text-align:center}.ui-layout__desktop-only{display:initial}.ui-layout__mobile-only{display:none}.ui-layout__bar-stack{display:flex;flex-direction:column;flex-shrink:0}.ui-layout__bar{display:flex;align-items:center;padding:0 var(--ui-spacing-4);font-size:var(--ui-font-size-sm)}.ui-layout__bar--notification{min-height:44px;gap:var(--ui-spacing-2);justify-content:center;font-weight:var(--ui-font-weight-medium)}.ui-layout__bar--notification-primary{background:var(--ui-color-primary);color:var(--ui-color-primary-contrast)}.ui-layout__bar--notification-accent{background:var(--ui-color-accent, var(--ui-color-primary));color:var(--ui-color-primary-contrast)}.ui-layout__bar--notification-warn{background:var(--ui-color-warn, #ef4444);color:#fff}.ui-layout__bar--notification-neutral{background:var(--ui-color-surface);color:var(--ui-color-text);border-bottom:1px solid var(--ui-color-border)}.ui-layout__bar--notification-ghost,.ui-layout__bar--notification-outline{background:var(--ui-color-bg-muted);color:var(--ui-color-text);border-bottom:1px solid var(--ui-color-border)}.ui-layout__bar--brand{min-height:56px;justify-content:space-between;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-4)}.ui-layout__bar--navigation{position:relative;min-height:44px;background:var(--ui-color-surface);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-1);overflow-x:auto;overflow-y:hidden;flex-wrap:nowrap;scrollbar-width:none;-webkit-overflow-scrolling:touch;mask-image:linear-gradient(to right,transparent 0,black var(--ui-spacing-4),black calc(100% - var(--ui-spacing-4)),transparent 100%)}.ui-layout__bar--navigation::-webkit-scrollbar{display:none}.ui-layout__bar--links{min-height:36px;background:var(--ui-color-bg-muted);border-bottom:1px solid var(--ui-color-border);gap:var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-layout__bar-text{flex:1;text-align:center}.ui-layout__bar-dismiss{display:flex;align-items:center;justify-content:center;background:none;border:none;cursor:pointer;padding:var(--ui-spacing-1);border-radius:var(--ui-radius-sm);color:inherit;opacity:.8;transition:opacity var(--ui-transition-fast)}.ui-layout__bar-dismiss:hover{opacity:1}.ui-layout__bar-dismiss:focus{outline:none}.ui-layout__bar-dismiss:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-brand-left{display:flex;align-items:center;gap:var(--ui-spacing-3);min-width:0}.ui-layout__bar-logo,.ui-layout__bar-logo-img{width:32px;height:32px;border-radius:var(--ui-radius-md);flex-shrink:0}.ui-layout__bar-logo{display:flex;align-items:center;justify-content:center;background:var(--ui-color-primary);color:var(--ui-color-primary-contrast);font-size:var(--ui-font-size-xs);font-weight:var(--ui-font-weight-bold)}.ui-layout__bar-logo-img{object-fit:contain}.ui-layout__bar-title{font-size:var(--ui-font-size-base);font-weight:var(--ui-font-weight-bold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-layout__topbar-hamburger{display:none}.ui-layout__user-dropdown{position:relative}.ui-layout__user-dropdown-trigger{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-1) var(--ui-spacing-2);background:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);cursor:pointer;font-family:inherit;font-size:var(--ui-font-size-sm);color:var(--ui-color-text);transition:all var(--ui-transition-fast)}.ui-layout__user-dropdown-trigger:hover{background:var(--ui-color-surface-hover)}.ui-layout__user-dropdown-trigger:focus{outline:none}.ui-layout__user-dropdown-trigger:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__user-avatar{width:28px;height:28px;border-radius:var(--ui-radius-full);object-fit:cover}.ui-layout__user-label{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:120px}.ui-layout__user-chevron{transition:transform var(--ui-transition-fast);color:var(--ui-color-text-muted)}[aria-expanded=true] .ui-layout__user-chevron{transform:rotate(180deg)}.ui-layout__user-dropdown-menu{position:absolute;top:calc(100% + var(--ui-spacing-1));right:0;min-width:180px;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-md);padding:var(--ui-spacing-1) 0;z-index:var(--ui-z-dropdown);animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}.ui-layout__user-dropdown-item{display:flex;align-items:center;gap:var(--ui-spacing-2);width:100%;padding:var(--ui-spacing-2) var(--ui-spacing-3);background:none;border:none;cursor:pointer;font-family:inherit;font-size:var(--ui-font-size-sm);color:var(--ui-color-text);text-decoration:none;text-align:left;transition:background var(--ui-transition-fast)}.ui-layout__user-dropdown-item:hover{background:var(--ui-color-surface-hover)}.ui-layout__user-dropdown-item:focus{outline:none}.ui-layout__user-dropdown-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__user-dropdown-divider{border:none;border-top:1px solid var(--ui-color-border);margin:var(--ui-spacing-1) 0}.ui-layout__hnav-link{display:flex;align-items:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-2) var(--ui-spacing-3);border-radius:var(--ui-radius-md);color:var(--ui-color-text-secondary);font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-medium);text-decoration:none;transition:all var(--ui-transition-fast);cursor:pointer;white-space:nowrap;background:none;border:none;font-family:inherit}.ui-layout__hnav-link:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__hnav-link:focus{outline:none}.ui-layout__hnav-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__hnav-link--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__hnav-link--parent{cursor:pointer}.ui-layout__hnav-link lucide-icon{transform:translateY(3px)}.ui-layout__hnav-chevron{transition:transform var(--ui-transition-fast)}.ui-layout__hnav-dropdown--open .ui-layout__hnav-chevron{transform:rotate(180deg)}.ui-layout__hnav-dropdown{position:relative}.ui-layout__hnav-dropdown-menu{position:absolute;top:calc(100% + var(--ui-spacing-1));left:0;min-width:200px;max-height:400px;overflow-y:auto;background:var(--ui-color-surface);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);box-shadow:var(--ui-shadow-md);padding:var(--ui-spacing-2);z-index:var(--ui-z-dropdown);animation:ui-layout-fade-in var(--ui-transition-fast) ease-out}.ui-layout__hnav-dropdown-label{display:block;font-size:.65rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;color:var(--ui-color-text-muted);padding:var(--ui-spacing-1) var(--ui-spacing-2);margin-top:var(--ui-spacing-2)}.ui-layout__hnav-dropdown-label:first-child{margin-top:0}.ui-layout__hnav-dropdown-item{display:block;padding:var(--ui-spacing-1) var(--ui-spacing-2);border-radius:var(--ui-radius-sm);font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);text-decoration:none;transition:all var(--ui-transition-fast)}.ui-layout__hnav-dropdown-item:hover{background:var(--ui-color-surface-hover);color:var(--ui-color-text)}.ui-layout__hnav-dropdown-item--active{color:var(--ui-color-primary);font-weight:var(--ui-font-weight-semibold)}.ui-layout__hnav-dropdown-item:focus{outline:none}.ui-layout__hnav-dropdown-item:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-link{display:inline-flex;align-items:center;gap:4px;color:var(--ui-color-text-muted);text-decoration:none;transition:color var(--ui-transition-fast)}.ui-layout__bar-link:hover{color:var(--ui-color-primary)}.ui-layout__bar-link:focus{outline:none}.ui-layout__bar-link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-layout__bar-link lucide-icon{transform:translateY(3px)}@media (max-width: 767px){.ui-layout__sidebar-footer{display:block}.ui-layout__topbar{display:flex}.ui-layout:not(.ui-layout--topbar){padding-top:56px}.ui-layout__backdrop,.ui-layout__backdrop--topbar{display:block}.ui-layout__sidebar{position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar--open{transform:translate(0)}.ui-layout__sidebar--collapsed{width:300px;min-width:300px}.ui-layout__sidebar--collapsed .ui-layout__sidebar-header{flex-direction:row;align-items:center;gap:0;padding:var(--ui-spacing-4)}.ui-layout__sidebar--topbar-drawer{display:flex}}@media (max-width: 767px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar{transition:none}}@media (max-width: 767px){.ui-layout__desktop-only{display:none!important}.ui-layout__mobile-only{display:initial!important}.ui-layout__content{padding:var(--ui-spacing-4)}.ui-layout__content--fullscreen{padding:0}.ui-layout__bottom-nav{display:flex}.ui-layout__fab{bottom:calc(var(--ui-spacing-4) + 56px + env(safe-area-inset-bottom,0))}.ui-layout__topbar-hamburger{display:flex}.ui-layout__bar--navigation,.ui-layout__bar--links,.ui-layout__user-label{display:none}.ui-layout__page-title{font-size:var(--ui-font-size-xl)}.ui-layout__breadcrumb{font-size:var(--ui-font-size-xs)}}@media (min-width: 768px) and (max-width: 1023px){.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer){position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer).ui-layout__sidebar--open{transform:translate(0)}}@media (min-width: 768px) and (max-width: 1023px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar:not(.ui-layout__sidebar--topbar-drawer){transition:none}}@media (min-width: 768px) and (max-width: 1023px){.ui-layout__topbar{display:flex}.ui-layout:not(.ui-layout--topbar){padding-top:56px}.ui-layout__backdrop,.ui-layout__backdrop--topbar{display:block}.ui-layout__desktop-only{display:none!important}.ui-layout__mobile-only{display:initial!important}.ui-layout__content{padding:var(--ui-spacing-6)}.ui-layout__content--fullscreen{padding:0}.ui-layout__topbar-hamburger{display:flex}.ui-layout__sidebar--topbar-drawer{display:flex;position:fixed;top:0;left:0;bottom:0;width:300px;min-width:300px;transform:translate(-100%);transition:transform var(--ui-transition-normal);z-index:var(--ui-z-modal)}.ui-layout__sidebar--topbar-drawer.ui-layout__sidebar--open{transform:translate(0)}}@media (min-width: 768px) and (max-width: 1023px) and (prefers-reduced-motion: reduce){.ui-layout__sidebar--topbar-drawer{transition:none}}\n"] }]
1720
+ }], propDecorators: { schema: [{
1721
+ type: Input,
1722
+ args: [{ required: true }]
1723
+ }], onEscape: [{
1724
+ type: HostListener,
1725
+ args: ['document:keydown.escape']
1726
+ }], onDocumentClick: [{
1727
+ type: HostListener,
1728
+ args: ['document:click', ['$event']]
1729
+ }] } });
1730
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout-builder.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/layout-builder/layout-builder.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Schema-driven application shell that materializes a full responsive layout\r\n * from a single `UiLayoutSchema` descriptor.\r\n *\r\n * Features:\r\n * - Two layout modes: sidebar (collapsible) and topbar (multi-bar stack)\r\n * - Programmatic mode switching with loader masking\r\n * - Content type: fluid, boxed, fullscreen\r\n * - Speed-dial FAB with context-sensitive actions\r\n * - Overlay loader with body-scroll locking\r\n * - Auto-derived breadcrumbs from navigation schema\r\n * - Skip-to-content link and full keyboard navigation\r\n *\r\n * @selector ui-layout-builder\r\n *\r\n * @example\r\n * ```html\r\n * <ui-layout-builder [schema]=\"layoutSchema\" />\r\n * ```\r\n */\r\nimport {\r\n  Component,\r\n  Input,\r\n  OnInit,\r\n  OnDestroy,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  ViewEncapsulation,\r\n  inject,\r\n  HostListener,\r\n} from '@angular/core';\r\nimport { Router, RouterOutlet, RouterLink, RouterLinkActive, NavigationEnd } from '@angular/router';\r\nimport { NgTemplateOutlet } from '@angular/common';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { Subscription, filter } from 'rxjs';\r\n\r\nimport { UiLayoutService } from './layout.service';\r\nimport {\r\n  UiLayoutSchema,\r\n  UiLayoutMode,\r\n  UiContentType,\r\n  UiLayoutBreadcrumb,\r\n  UiNavItem,\r\n  UiNavSection,\r\n  UiLayoutFabConfig,\r\n  UiFabAction,\r\n  UiBarConfig,\r\n} from './layout-builder.types';\r\n\r\n@Component({\r\n  selector: 'ui-layout-builder',\r\n  standalone: true,\r\n  imports: [\r\n    RouterOutlet,\r\n    RouterLink,\r\n    RouterLinkActive,\r\n    NgTemplateOutlet,\r\n    LucideAngularModule,\r\n  ],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: { class: 'ui-layout-builder-host' },\r\n  template: `\r\n    <!-- Skip to content (a11y) -->\r\n    <a class=\"ui-layout__skip-link\" href=\"#ui-layout-main\">\r\n      Vai al contenuto principale\r\n    </a>\r\n\r\n    <div\r\n      class=\"ui-layout\"\r\n      [class.ui-layout--sidebar-collapsed]=\"layoutMode === 'sidebar' && sidebarCollapsed\"\r\n      [class.ui-layout--topbar]=\"layoutMode === 'topbar'\"\r\n    >\r\n      <!-- ─── Overlay Loader ─────────────────────────────────────── -->\r\n      @if (loading) {\r\n        <div\r\n          class=\"ui-layout__loader-overlay\"\r\n          role=\"alert\"\r\n          aria-live=\"assertive\"\r\n          aria-label=\"Caricamento in corso\"\r\n        >\r\n          <div class=\"ui-layout__loader-spinner\" aria-hidden=\"true\">\r\n            <svg viewBox=\"0 0 50 50\" class=\"ui-layout__loader-svg\">\r\n              <circle cx=\"25\" cy=\"25\" r=\"20\" fill=\"none\" stroke-width=\"4\" />\r\n            </svg>\r\n          </div>\r\n        </div>\r\n      }\r\n\r\n      @switch (layoutMode) {\r\n        <!-- ════════════════════════════════════════════════════════ -->\r\n        <!-- ═══ SIDEBAR MODE ═════════════════════════════════════ -->\r\n        <!-- ════════════════════════════════════════════════════════ -->\r\n        @case ('sidebar') {\r\n          <!-- Mobile Top Bar -->\r\n          <header class=\"ui-layout__topbar\">\r\n            <button\r\n              class=\"ui-layout__hamburger\"\r\n              (click)=\"layoutService.openMobileSidebar()\"\r\n              aria-label=\"Apri navigazione\"\r\n            >\r\n              <span class=\"ui-layout__hamburger-line\"></span>\r\n              <span class=\"ui-layout__hamburger-line\"></span>\r\n              <span class=\"ui-layout__hamburger-line\"></span>\r\n            </button>\r\n            @if (schema.header.logo) {\r\n              @if (schema.header.logo.src) {\r\n                <img\r\n                  class=\"ui-layout__topbar-logo-img\"\r\n                  [src]=\"schema.header.logo.src\"\r\n                  [alt]=\"schema.header.logo.alt || schema.header.title\"\r\n                />\r\n              } @else if (schema.header.logo.text) {\r\n                <span class=\"ui-layout__topbar-logo\">{{ schema.header.logo.text }}</span>\r\n              }\r\n            }\r\n            <span class=\"ui-layout__topbar-title\">{{ schema.header.title }}</span>\r\n          </header>\r\n\r\n          <!-- Mobile Backdrop -->\r\n          @if (sidebarOpen) {\r\n            <div\r\n              class=\"ui-layout__backdrop\"\r\n              (click)=\"layoutService.closeMobileSidebar()\"\r\n              aria-hidden=\"true\"\r\n            ></div>\r\n          }\r\n\r\n          <!-- Body (sidebar + content) -->\r\n          <div class=\"ui-layout__body\">\r\n            <aside\r\n              class=\"ui-layout__sidebar\"\r\n              [class.ui-layout__sidebar--collapsed]=\"sidebarCollapsed\"\r\n              [class.ui-layout__sidebar--open]=\"sidebarOpen\"\r\n              role=\"navigation\"\r\n              aria-label=\"Navigazione principale\"\r\n            >\r\n              <div class=\"ui-layout__sidebar-header\">\r\n                <div class=\"ui-layout__sidebar-brand\">\r\n                  @if (schema.header.logo) {\r\n                    @if (schema.header.logo.src) {\r\n                      <img\r\n                        class=\"ui-layout__sidebar-logo-img\"\r\n                        [src]=\"schema.header.logo.src\"\r\n                        [alt]=\"schema.header.logo.alt || schema.header.title\"\r\n                      />\r\n                    } @else if (schema.header.logo.text) {\r\n                      <span class=\"ui-layout__sidebar-logo\">{{ schema.header.logo.text }}</span>\r\n                    }\r\n                  }\r\n                  @if (!sidebarCollapsed || sidebarOpen) {\r\n                    <span class=\"ui-layout__sidebar-title\">{{ schema.header.title }}</span>\r\n                  }\r\n                </div>\r\n\r\n                @if (schema.navigation.collapsible !== false) {\r\n                  <button\r\n                    class=\"ui-layout__collapse-btn ui-layout__desktop-only\"\r\n                    (click)=\"layoutService.toggleSidebar()\"\r\n                    [attr.aria-label]=\"sidebarCollapsed ? 'Espandi sidebar' : 'Comprimi sidebar'\"\r\n                  >\r\n                    <lucide-icon\r\n                      name=\"chevrons-left\"\r\n                      [size]=\"16\"\r\n                      class=\"ui-layout__collapse-icon\"\r\n                      aria-hidden=\"true\"\r\n                    />\r\n                  </button>\r\n                }\r\n\r\n                <button\r\n                  class=\"ui-layout__close-btn ui-layout__mobile-only\"\r\n                  (click)=\"layoutService.closeMobileSidebar()\"\r\n                  aria-label=\"Chiudi navigazione\"\r\n                >\r\n                  <lucide-icon name=\"x\" [size]=\"16\" aria-hidden=\"true\" />\r\n                </button>\r\n              </div>\r\n\r\n              @if (!sidebarCollapsed || sidebarOpen) {\r\n                <nav class=\"ui-layout__sidebar-nav\">\r\n                  @if (schema.navigation.sections?.length) {\r\n                    @for (section of schema.navigation.sections; track section.id) {\r\n                      <div class=\"ui-layout__nav-section\">\r\n                        <span class=\"ui-layout__nav-section-title\">{{ section.label }}</span>\r\n                        @for (item of getItemsForSection(section); track item.id) {\r\n                          @if (!isItemHidden(item)) {\r\n                            <ng-container\r\n                              *ngTemplateOutlet=\"navItemTpl; context: { $implicit: item }\"\r\n                            />\r\n                          }\r\n                        }\r\n                      </div>\r\n                    }\r\n                  } @else {\r\n                    <div class=\"ui-layout__nav-section\">\r\n                      @for (item of schema.navigation.items; track item.id) {\r\n                        @if (!isItemHidden(item)) {\r\n                          <ng-container\r\n                            *ngTemplateOutlet=\"navItemTpl; context: { $implicit: item }\"\r\n                          />\r\n                        }\r\n                      }\r\n                    </div>\r\n                  }\r\n                </nav>\r\n              }\r\n            </aside>\r\n\r\n            <!-- Main content area -->\r\n            <main\r\n              class=\"ui-layout__content\"\r\n              [class.ui-layout__content--boxed]=\"contentType === 'boxed'\"\r\n              [class.ui-layout__content--fullscreen]=\"contentType === 'fullscreen'\"\r\n              [class.ui-layout__content--transitioning]=\"contentTransitioning\"\r\n              (animationend)=\"contentTransitioning = false\"\r\n              id=\"ui-layout-main\"\r\n              tabindex=\"-1\"\r\n            >\r\n              <ng-container *ngTemplateOutlet=\"pageHeaderTpl\" />\r\n              <router-outlet />\r\n            </main>\r\n          </div>\r\n        }\r\n\r\n        <!-- ════════════════════════════════════════════════════════ -->\r\n        <!-- ═══ TOPBAR MODE ══════════════════════════════════════ -->\r\n        <!-- ════════════════════════════════════════════════════════ -->\r\n        @case ('topbar') {\r\n          <!-- Bar stack -->\r\n          <div class=\"ui-layout__bar-stack\">\r\n            @for (bar of visibleBars; track bar.id) {\r\n              @switch (bar.type) {\r\n                @case ('notification') {\r\n                  @if (bar.notification) {\r\n                    <div\r\n                      class=\"ui-layout__bar ui-layout__bar--notification\"\r\n                      [class]=\"'ui-layout__bar--notification-' + (bar.notification.variant || 'primary')\"\r\n                      role=\"alert\"\r\n                    >\r\n                      @if (bar.notification.icon) {\r\n                        <lucide-icon [name]=\"bar.notification.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                      }\r\n                      <span class=\"ui-layout__bar-text\">{{ bar.notification.text }}</span>\r\n                      @if (bar.notification.dismissible !== false) {\r\n                        <button\r\n                          class=\"ui-layout__bar-dismiss\"\r\n                          (click)=\"dismissBar(bar.id)\"\r\n                          aria-label=\"Chiudi notifica\"\r\n                        >\r\n                          <lucide-icon name=\"x\" [size]=\"14\" aria-hidden=\"true\" />\r\n                        </button>\r\n                      }\r\n                    </div>\r\n                  }\r\n                }\r\n\r\n                @case ('brand') {\r\n                  @if (bar.brand; as brand) {\r\n                    <div class=\"ui-layout__bar ui-layout__bar--brand\">\r\n                      <div class=\"ui-layout__bar-brand-left\">\r\n                        <button\r\n                          class=\"ui-layout__hamburger ui-layout__topbar-hamburger\"\r\n                          (click)=\"layoutService.openMobileSidebar()\"\r\n                          aria-label=\"Apri navigazione\"\r\n                        >\r\n                          <span class=\"ui-layout__hamburger-line\"></span>\r\n                          <span class=\"ui-layout__hamburger-line\"></span>\r\n                          <span class=\"ui-layout__hamburger-line\"></span>\r\n                        </button>\r\n                        @if (brand.logo; as logo) {\r\n                          @if (logo.src) {\r\n                            <img\r\n                              class=\"ui-layout__bar-logo-img\"\r\n                              [src]=\"logo.src\"\r\n                              [alt]=\"logo.alt || brand.title || ''\"\r\n                            />\r\n                          } @else if (logo.text) {\r\n                            <span class=\"ui-layout__bar-logo\">{{ logo.text }}</span>\r\n                          }\r\n                        }\r\n                        @if (brand.title; as brandTitle) {\r\n                          <span class=\"ui-layout__bar-title\">{{ brandTitle }}</span>\r\n                        }\r\n                      </div>\r\n                      @if (brand.userDropdown; as dropdown) {\r\n                        <div class=\"ui-layout__user-dropdown\">\r\n                          <button\r\n                            class=\"ui-layout__user-dropdown-trigger\"\r\n                            (click)=\"userDropdownOpen = !userDropdownOpen\"\r\n                            [attr.aria-expanded]=\"userDropdownOpen\"\r\n                            aria-haspopup=\"true\"\r\n                          >\r\n                            @if (dropdown.avatar?.src) {\r\n                              <img\r\n                                class=\"ui-layout__user-avatar\"\r\n                                [src]=\"dropdown.avatar!.src\"\r\n                                [alt]=\"dropdown.avatar!.alt || dropdown.label\"\r\n                              />\r\n                            } @else {\r\n                              <lucide-icon\r\n                                [name]=\"dropdown.icon || 'user'\"\r\n                                [size]=\"18\"\r\n                                aria-hidden=\"true\"\r\n                              />\r\n                            }\r\n                            <span class=\"ui-layout__user-label\">{{ dropdown.label }}</span>\r\n                            <lucide-icon name=\"chevron-down\" [size]=\"14\" aria-hidden=\"true\" class=\"ui-layout__user-chevron\" />\r\n                          </button>\r\n                          @if (userDropdownOpen) {\r\n                            <div class=\"ui-layout__user-dropdown-menu\" role=\"menu\">\r\n                              @for (dItem of dropdown.items; track dItem.id) {\r\n                                @if (dItem.divider) {\r\n                                  <hr class=\"ui-layout__user-dropdown-divider\" />\r\n                                }\r\n                                @if (dItem.route) {\r\n                                  <a\r\n                                    class=\"ui-layout__user-dropdown-item\"\r\n                                    [routerLink]=\"dItem.route\"\r\n                                    role=\"menuitem\"\r\n                                    (click)=\"userDropdownOpen = false\"\r\n                                  >\r\n                                    @if (dItem.icon) {\r\n                                      <lucide-icon [name]=\"dItem.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                                    }\r\n                                    {{ dItem.label }}\r\n                                  </a>\r\n                                } @else {\r\n                                  <button\r\n                                    class=\"ui-layout__user-dropdown-item\"\r\n                                    role=\"menuitem\"\r\n                                    (click)=\"onDropdownAction(dItem)\"\r\n                                  >\r\n                                    @if (dItem.icon) {\r\n                                      <lucide-icon [name]=\"dItem.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                                    }\r\n                                    {{ dItem.label }}\r\n                                  </button>\r\n                                }\r\n                              }\r\n                            </div>\r\n                          }\r\n                        </div>\r\n                      }\r\n                    </div>\r\n                  }\r\n                }\r\n\r\n                @case ('navigation') {\r\n                  <nav class=\"ui-layout__bar ui-layout__bar--navigation\" role=\"navigation\" aria-label=\"Navigazione principale\">\r\n                    @for (item of getTopbarNavItems(bar); track item.id) {\r\n                      @if (!isItemHidden(item)) {\r\n                        @if (item.children?.length) {\r\n                          <div class=\"ui-layout__hnav-dropdown\" [class.ui-layout__hnav-dropdown--open]=\"hnavDropdownId === item.id\">\r\n                            <button\r\n                              class=\"ui-layout__hnav-link ui-layout__hnav-link--parent\"\r\n                              [class.ui-layout__hnav-link--active]=\"isParentActive(item)\"\r\n                              (click)=\"toggleHnavDropdown(item.id)\"\r\n                              [attr.aria-expanded]=\"hnavDropdownId === item.id\"\r\n                            >\r\n                              @if (item.icon) {\r\n                                <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                              }\r\n                              <span>{{ item.label }}</span>\r\n                              <lucide-icon name=\"chevron-down\" [size]=\"14\" aria-hidden=\"true\" class=\"ui-layout__hnav-chevron\" />\r\n                            </button>\r\n                            @if (hnavDropdownId === item.id) {\r\n                              <div class=\"ui-layout__hnav-dropdown-menu\">\r\n                                @for (group of item.children; track group.label) {\r\n                                  <span class=\"ui-layout__hnav-dropdown-label\">{{ group.label }}</span>\r\n                                  @for (child of group.items; track child.id) {\r\n                                    @if (!isItemHidden(child)) {\r\n                                      <a\r\n                                        class=\"ui-layout__hnav-dropdown-item\"\r\n                                        [routerLink]=\"child.route\"\r\n                                        routerLinkActive=\"ui-layout__hnav-dropdown-item--active\"\r\n                                        (click)=\"hnavDropdownId = null; onNavClick()\"\r\n                                      >\r\n                                        {{ child.label }}\r\n                                      </a>\r\n                                    }\r\n                                  }\r\n                                }\r\n                              </div>\r\n                            }\r\n                          </div>\r\n                        } @else if (item.type === 'external') {\r\n                          <a\r\n                            class=\"ui-layout__hnav-link\"\r\n                            [href]=\"item.href\"\r\n                            [target]=\"item.target || '_blank'\"\r\n                            [attr.rel]=\"item.target === '_blank' ? 'noopener noreferrer' : null\"\r\n                          >\r\n                            @if (item.icon) {\r\n                              <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                            }\r\n                            <span>{{ item.label }}</span>\r\n                            <lucide-icon name=\"external-link\" [size]=\"12\" aria-hidden=\"true\" />\r\n                          </a>\r\n                        } @else if (item.type !== 'divider') {\r\n                          <a\r\n                            class=\"ui-layout__hnav-link\"\r\n                            [routerLink]=\"item.route\"\r\n                            routerLinkActive=\"ui-layout__hnav-link--active\"\r\n                            [routerLinkActiveOptions]=\"{ exact: item.routeActiveExact || false }\"\r\n                          >\r\n                            @if (item.icon) {\r\n                              <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" />\r\n                            }\r\n                            <span>{{ item.label }}</span>\r\n                            @if (getItemBadge(item)) {\r\n                              <span class=\"ui-layout__badge\">{{ getItemBadge(item) }}</span>\r\n                            }\r\n                          </a>\r\n                        }\r\n                      }\r\n                    }\r\n                  </nav>\r\n                }\r\n\r\n                @case ('links') {\r\n                  @if (bar.links) {\r\n                    <div\r\n                      class=\"ui-layout__bar ui-layout__bar--links\"\r\n                      [style.justify-content]=\"bar.links.align === 'start' ? 'flex-start' : bar.links.align === 'center' ? 'center' : bar.links.align === 'space-between' ? 'space-between' : 'flex-end'\"\r\n                    >\r\n                      @for (lnk of bar.links.items; track lnk.label || lnk.ariaLabel || $index) {\r\n                        @if (lnk.route) {\r\n                          <a class=\"ui-layout__bar-link\" [routerLink]=\"lnk.route\" [attr.aria-label]=\"lnk.ariaLabel || null\">\r\n                            @if (lnk.icon) {\r\n                              <lucide-icon [name]=\"lnk.icon\" [size]=\"14\" aria-hidden=\"true\" />\r\n                            }\r\n                            @if (lnk.label) {\r\n                              <span>{{ lnk.label }}</span>\r\n                            }\r\n                          </a>\r\n                        } @else if (lnk.href) {\r\n                          <a\r\n                            class=\"ui-layout__bar-link\"\r\n                            [href]=\"lnk.href\"\r\n                            [target]=\"lnk.target || '_self'\"\r\n                            [attr.rel]=\"lnk.target === '_blank' ? 'noopener noreferrer' : null\"\r\n                            [attr.aria-label]=\"lnk.ariaLabel || null\"\r\n                          >\r\n                            @if (lnk.icon) {\r\n                              <lucide-icon [name]=\"lnk.icon\" [size]=\"14\" aria-hidden=\"true\" />\r\n                            }\r\n                            @if (lnk.label) {\r\n                              <span>{{ lnk.label }}</span>\r\n                            }\r\n                          </a>\r\n                        }\r\n                      }\r\n                    </div>\r\n                  }\r\n                }\r\n              }\r\n            }\r\n          </div>\r\n\r\n          <!-- Mobile Backdrop (topbar mode) -->\r\n          @if (sidebarOpen) {\r\n            <div\r\n              class=\"ui-layout__backdrop ui-layout__backdrop--topbar\"\r\n              (click)=\"layoutService.closeMobileSidebar()\"\r\n              aria-hidden=\"true\"\r\n            ></div>\r\n          }\r\n\r\n          <!-- Mobile drawer (topbar mode) -->\r\n          <aside\r\n            class=\"ui-layout__sidebar ui-layout__sidebar--topbar-drawer\"\r\n            [class.ui-layout__sidebar--open]=\"sidebarOpen\"\r\n            role=\"navigation\"\r\n            aria-label=\"Navigazione principale\"\r\n          >\r\n            <div class=\"ui-layout__sidebar-header\">\r\n              <div class=\"ui-layout__sidebar-brand\">\r\n                @if (schema.header.logo) {\r\n                  @if (schema.header.logo.src) {\r\n                    <img class=\"ui-layout__sidebar-logo-img\" [src]=\"schema.header.logo.src\" [alt]=\"schema.header.logo.alt || schema.header.title\" />\r\n                  } @else if (schema.header.logo.text) {\r\n                    <span class=\"ui-layout__sidebar-logo\">{{ schema.header.logo.text }}</span>\r\n                  }\r\n                }\r\n                <span class=\"ui-layout__sidebar-title\">{{ schema.header.title }}</span>\r\n              </div>\r\n              <button class=\"ui-layout__close-btn\" (click)=\"layoutService.closeMobileSidebar()\" aria-label=\"Chiudi navigazione\">\r\n                <lucide-icon name=\"x\" [size]=\"16\" aria-hidden=\"true\" />\r\n              </button>\r\n            </div>\r\n            <nav class=\"ui-layout__sidebar-nav\">\r\n              @if (schema.navigation.sections?.length) {\r\n                @for (section of schema.navigation.sections; track section.id) {\r\n                  <div class=\"ui-layout__nav-section\">\r\n                    <span class=\"ui-layout__nav-section-title\">{{ section.label }}</span>\r\n                    @for (item of getItemsForSection(section); track item.id) {\r\n                      @if (!isItemHidden(item)) {\r\n                        <ng-container *ngTemplateOutlet=\"navItemTpl; context: { $implicit: item }\" />\r\n                      }\r\n                    }\r\n                  </div>\r\n                }\r\n              } @else {\r\n                <div class=\"ui-layout__nav-section\">\r\n                  @for (item of schema.navigation.items; track item.id) {\r\n                    @if (!isItemHidden(item)) {\r\n                      <ng-container *ngTemplateOutlet=\"navItemTpl; context: { $implicit: item }\" />\r\n                    }\r\n                  }\r\n                </div>\r\n              }\r\n            </nav>\r\n          </aside>\r\n\r\n          <!-- Main content area (topbar mode) -->\r\n          <main\r\n            class=\"ui-layout__content ui-layout__content--topbar\"\r\n            [class.ui-layout__content--boxed]=\"contentType === 'boxed'\"\r\n            [class.ui-layout__content--fullscreen]=\"contentType === 'fullscreen'\"\r\n            [class.ui-layout__content--transitioning]=\"contentTransitioning\"\r\n            (animationend)=\"contentTransitioning = false\"\r\n            id=\"ui-layout-main\"\r\n            tabindex=\"-1\"\r\n          >\r\n            <ng-container *ngTemplateOutlet=\"pageHeaderTpl\" />\r\n            <router-outlet />\r\n          </main>\r\n        }\r\n      }\r\n\r\n      <!-- ─── Footer ─────────────────────────────────────────────── -->\r\n      @if (schema.footer) {\r\n        <footer class=\"ui-layout__footer\">\r\n          @if (schema.footer.links?.length) {\r\n            <nav class=\"ui-layout__footer-links\" aria-label=\"Link footer\">\r\n              @for (link of schema.footer.links; track link.label) {\r\n                @if (link.route) {\r\n                  <a class=\"ui-layout__footer-link\" [routerLink]=\"link.route\">\r\n                    @if (link.icon) {\r\n                      <lucide-icon [name]=\"link.icon\" [size]=\"14\" aria-hidden=\"true\" />\r\n                    }\r\n                    {{ link.label }}\r\n                  </a>\r\n                } @else if (link.href) {\r\n                  <a\r\n                    class=\"ui-layout__footer-link\"\r\n                    [href]=\"link.href\"\r\n                    [target]=\"link.target || '_self'\"\r\n                    [attr.rel]=\"link.target === '_blank' ? 'noopener noreferrer' : null\"\r\n                  >\r\n                    @if (link.icon) {\r\n                      <lucide-icon [name]=\"link.icon\" [size]=\"14\" aria-hidden=\"true\" />\r\n                    }\r\n                    {{ link.label }}\r\n                  </a>\r\n                }\r\n              }\r\n            </nav>\r\n          }\r\n          @if (schema.footer.text) {\r\n            <span class=\"ui-layout__footer-text\">{{ schema.footer.text }}</span>\r\n          }\r\n        </footer>\r\n      }\r\n\r\n      <!-- ─── Speed Dial FAB ─────────────────────────────────────── -->\r\n      @if (currentFabConfig) {\r\n        <div\r\n          class=\"ui-layout__fab\"\r\n          [class.ui-layout__fab--open]=\"fabOpen\"\r\n          [class.ui-layout__fab--bottom-left]=\"currentFabConfig.position === 'bottom-left'\"\r\n          role=\"complementary\"\r\n          aria-label=\"Azioni rapide\"\r\n        >\r\n          @if (fabOpen && currentFabConfig.secondaryActions?.length) {\r\n            <div class=\"ui-layout__fab-actions\">\r\n              @for (action of currentFabConfig.secondaryActions; track action.id; let i = $index) {\r\n                <button\r\n                  class=\"ui-layout__fab-action\"\r\n                  [class]=\"'ui-layout__fab-action--' + (action.variant || 'neutral')\"\r\n                  [attr.aria-label]=\"action.ariaLabel || action.label\"\r\n                  [title]=\"action.label\"\r\n                  [style.animation-delay]=\"(i * 50) + 'ms'\"\r\n                  (click)=\"onFabAction(action)\"\r\n                >\r\n                  <lucide-icon [name]=\"action.icon\" [size]=\"18\" aria-hidden=\"true\" />\r\n                  <span class=\"ui-layout__fab-action-label\">{{ action.label }}</span>\r\n                </button>\r\n              }\r\n            </div>\r\n          }\r\n          <button\r\n            class=\"ui-layout__fab-main\"\r\n            [class]=\"'ui-layout__fab-main--' + (currentFabConfig.mainAction.variant || 'primary')\"\r\n            [attr.aria-label]=\"currentFabConfig.mainAction.ariaLabel || currentFabConfig.mainAction.label\"\r\n            [attr.aria-expanded]=\"currentFabConfig.secondaryActions?.length ? fabOpen : null\"\r\n            (click)=\"onFabMainClick()\"\r\n          >\r\n            <lucide-icon\r\n              [name]=\"currentFabConfig.mainAction.icon\"\r\n              [size]=\"24\"\r\n              aria-hidden=\"true\"\r\n              class=\"ui-layout__fab-main-icon\"\r\n            />\r\n          </button>\r\n        </div>\r\n      }\r\n\r\n      <!-- ─── Bottom Navigation (mobile, sidebar mode) ────────────── -->\r\n      @if (layoutMode === 'sidebar' && bottomNavItems.length > 0 && mobileMode !== 'drawer') {\r\n        <nav\r\n          class=\"ui-layout__bottom-nav\"\r\n          role=\"navigation\"\r\n          aria-label=\"Navigazione rapida\"\r\n        >\r\n          @for (item of bottomNavItems; track item.id) {\r\n            <a\r\n              class=\"ui-layout__bottom-nav-item\"\r\n              [routerLink]=\"item.route\"\r\n              routerLinkActive=\"ui-layout__bottom-nav-item--active\"\r\n              [routerLinkActiveOptions]=\"{ exact: item.routeActiveExact || false }\"\r\n              [attr.aria-current]=\"isRouteActive(item) ? 'page' : null\"\r\n            >\r\n              @if (item.icon) {\r\n                <lucide-icon [name]=\"item.icon\" [size]=\"20\" aria-hidden=\"true\" />\r\n              }\r\n              <span class=\"ui-layout__bottom-nav-label\">{{ item.label }}</span>\r\n              @if (getItemBadge(item)) {\r\n                <span class=\"ui-layout__badge\" aria-label=\"Notifica\">{{ getItemBadge(item) }}</span>\r\n              }\r\n            </a>\r\n          }\r\n        </nav>\r\n      }\r\n    </div>\r\n\r\n    <!-- ═══ SHARED TEMPLATES ═══════════════════════════════════════ -->\r\n\r\n    <!-- Page header with auto-derived breadcrumbs -->\r\n    <ng-template #pageHeaderTpl>\r\n      @if (schema.pageHeader?.show !== false && breadcrumbs.length > 0) {\r\n        <header class=\"ui-layout__page-header\">\r\n          <nav aria-label=\"Breadcrumb\" class=\"ui-layout__breadcrumb-nav\">\r\n            <ol class=\"ui-layout__breadcrumb\" itemscope itemtype=\"https://schema.org/BreadcrumbList\">\r\n              @if (schema.pageHeader?.showHome !== false) {\r\n                <li class=\"ui-layout__breadcrumb-item\" itemprop=\"itemListElement\" itemscope itemtype=\"https://schema.org/ListItem\">\r\n                  <a itemprop=\"item\" [routerLink]=\"schema.pageHeader?.homeRoute || '/'\" class=\"ui-layout__breadcrumb-link\" aria-label=\"Home\">\r\n                    <lucide-icon name=\"home\" [size]=\"14\" aria-hidden=\"true\" />\r\n                    <span itemprop=\"name\" class=\"ui-layout__breadcrumb-text\">Home</span>\r\n                  </a>\r\n                  <meta itemprop=\"position\" content=\"1\" />\r\n                  <span class=\"ui-layout__breadcrumb-sep\" aria-hidden=\"true\">\r\n                    <lucide-icon name=\"chevron-right\" [size]=\"14\" />\r\n                  </span>\r\n                </li>\r\n              }\r\n              @for (crumb of breadcrumbs; track crumb.label; let i = $index) {\r\n                <li\r\n                  class=\"ui-layout__breadcrumb-item\"\r\n                  [class.ui-layout__breadcrumb-item--current]=\"crumb.isLast\"\r\n                  itemprop=\"itemListElement\"\r\n                  itemscope\r\n                  itemtype=\"https://schema.org/ListItem\"\r\n                >\r\n                  @if (!crumb.isLast && crumb.url) {\r\n                    <a itemprop=\"item\" [routerLink]=\"crumb.url\" class=\"ui-layout__breadcrumb-link\">\r\n                      <span itemprop=\"name\" class=\"ui-layout__breadcrumb-text\">{{ crumb.label }}</span>\r\n                    </a>\r\n                    <span class=\"ui-layout__breadcrumb-sep\" aria-hidden=\"true\">\r\n                      <lucide-icon name=\"chevron-right\" [size]=\"14\" />\r\n                    </span>\r\n                  } @else {\r\n                    <span itemprop=\"name\" class=\"ui-layout__breadcrumb-text ui-layout__breadcrumb-text--current\" aria-current=\"page\">\r\n                      {{ crumb.label }}\r\n                    </span>\r\n                  }\r\n                  <meta itemprop=\"position\" [attr.content]=\"(schema.pageHeader?.showHome !== false) ? i + 2 : i + 1\" />\r\n                </li>\r\n              }\r\n            </ol>\r\n          </nav>\r\n          <h1 class=\"ui-layout__page-title\">{{ pageTitle }}</h1>\r\n        </header>\r\n      }\r\n    </ng-template>\r\n\r\n    <!-- Nav item template (sidebar, recursive-friendly) -->\r\n    <ng-template #navItemTpl let-item>\r\n      @if (item.type === 'divider') {\r\n        <hr class=\"ui-layout__nav-divider\" />\r\n      } @else if (item.type === 'external') {\r\n        <a\r\n          class=\"ui-layout__nav-link\"\r\n          [href]=\"item.href\"\r\n          [target]=\"item.target || '_blank'\"\r\n          [attr.rel]=\"item.target === '_blank' ? 'noopener noreferrer' : null\"\r\n          (click)=\"onNavClick()\"\r\n        >\r\n          @if (item.icon) {\r\n            <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" class=\"ui-layout__nav-icon\" />\r\n          }\r\n          <span class=\"ui-layout__nav-label\">{{ item.label }}</span>\r\n          <lucide-icon name=\"external-link\" [size]=\"12\" aria-hidden=\"true\" class=\"ui-layout__nav-external\" />\r\n          @if (getItemBadge(item)) {\r\n            <span class=\"ui-layout__badge\">{{ getItemBadge(item) }}</span>\r\n          }\r\n        </a>\r\n      } @else if (item.children?.length) {\r\n        <button\r\n          class=\"ui-layout__nav-link ui-layout__nav-link--parent\"\r\n          [class.ui-layout__nav-link--active]=\"isParentActive(item)\"\r\n          [class.ui-layout__nav-link--expanded]=\"expandedIds.has(item.id)\"\r\n          [attr.aria-expanded]=\"expandedIds.has(item.id)\"\r\n          (click)=\"toggleExpand(item)\"\r\n        >\r\n          @if (item.icon) {\r\n            <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" class=\"ui-layout__nav-icon\" />\r\n          }\r\n          <span class=\"ui-layout__nav-label\">{{ item.label }}</span>\r\n          @if (getItemBadge(item)) {\r\n            <span class=\"ui-layout__badge\">{{ getItemBadge(item) }}</span>\r\n          }\r\n          <span class=\"ui-layout__nav-chevron\" aria-hidden=\"true\">\r\n            <lucide-icon name=\"chevron-right\" [size]=\"14\" />\r\n          </span>\r\n        </button>\r\n        @if (expandedIds.has(item.id)) {\r\n          <div class=\"ui-layout__nav-children\">\r\n            @for (group of item.children; track group.label) {\r\n              <span class=\"ui-layout__nav-group-label\">{{ group.label }}</span>\r\n              @for (child of group.items; track child.id) {\r\n                @if (!isItemHidden(child)) {\r\n                  <a\r\n                    class=\"ui-layout__nav-link ui-layout__nav-link--child\"\r\n                    [routerLink]=\"child.route\"\r\n                    routerLinkActive=\"ui-layout__nav-link--active\"\r\n                    (click)=\"onNavClick()\"\r\n                  >\r\n                    <span class=\"ui-layout__nav-label\">{{ child.label }}</span>\r\n                    @if (getItemBadge(child)) {\r\n                      <span class=\"ui-layout__badge\">{{ getItemBadge(child) }}</span>\r\n                    }\r\n                  </a>\r\n                }\r\n              }\r\n            }\r\n          </div>\r\n        }\r\n      } @else {\r\n        <a\r\n          class=\"ui-layout__nav-link\"\r\n          [routerLink]=\"item.route\"\r\n          routerLinkActive=\"ui-layout__nav-link--active\"\r\n          [routerLinkActiveOptions]=\"{ exact: item.routeActiveExact || false }\"\r\n          [attr.aria-current]=\"isRouteActive(item) ? 'page' : null\"\r\n          (click)=\"onNavClick()\"\r\n        >\r\n          @if (item.icon) {\r\n            <lucide-icon [name]=\"item.icon\" [size]=\"16\" aria-hidden=\"true\" class=\"ui-layout__nav-icon\" />\r\n          }\r\n          <span class=\"ui-layout__nav-label\">{{ item.label }}</span>\r\n          @if (getItemBadge(item)) {\r\n            <span class=\"ui-layout__badge\">{{ getItemBadge(item) }}</span>\r\n          }\r\n        </a>\r\n      }\r\n    </ng-template>\r\n  `,\r\n  styleUrl: './layout-builder.component.scss',\r\n})\r\nexport class UiLayoutBuilderComponent implements OnInit, OnDestroy {\r\n  /** Full layout descriptor. */\r\n  @Input({ required: true }) schema!: UiLayoutSchema;\r\n\r\n  readonly layoutService = inject(UiLayoutService);\r\n  private readonly router = inject(Router);\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n  private subs: Subscription[] = [];\r\n\r\n  currentUrl = '';\r\n  expandedIds = new Set<string>();\r\n  fabOpen = false;\r\n  currentFabConfig: UiLayoutFabConfig | null = null;\r\n\r\n  loading = false;\r\n  sidebarOpen = false;\r\n  sidebarCollapsed = false;\r\n  layoutMode: UiLayoutMode = 'sidebar';\r\n  contentType: UiContentType = 'fluid';\r\n  contentTransitioning = false;\r\n  breadcrumbs: UiLayoutBreadcrumb[] = [];\r\n  userDropdownOpen = false;\r\n  hnavDropdownId: string | null = null;\r\n\r\n  private navBadges = new Map<string, string | number | null>();\r\n  private navVisibility = new Map<string, boolean>();\r\n  private barVisibility = new Map<string, boolean>();\r\n\r\n  /** Resolved page title from the last breadcrumb. */\r\n  get pageTitle(): string {\r\n    if (!this.breadcrumbs.length) return '';\r\n    return this.breadcrumbs[this.breadcrumbs.length - 1].label;\r\n  }\r\n\r\n  /** Items flagged for the mobile bottom navigation bar (max 5). */\r\n  get bottomNavItems(): UiNavItem[] {\r\n    return this.schema.navigation.items\r\n      .filter((i) => i.bottomNav && !this.isItemHidden(i) && i.route)\r\n      .sort((a, b) => (a.bottomNavOrder ?? 99) - (b.bottomNavOrder ?? 99))\r\n      .slice(0, 5);\r\n  }\r\n\r\n  get mobileMode(): string {\r\n    return this.schema.navigation.mobileMode || 'both';\r\n  }\r\n\r\n  /** Topbar bars filtered by visibility (schema default + service overrides). */\r\n  get visibleBars(): UiBarConfig[] {\r\n    if (!this.schema.topbar?.bars) return [];\r\n    return this.schema.topbar.bars.filter((bar) =>\r\n      this.layoutService.isBarVisible(bar.id, bar.visible !== false),\r\n    );\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.currentUrl = this.router.url;\r\n    this.autoExpandFromUrl(this.currentUrl);\r\n\r\n    if (this.schema.mode) {\r\n      this.layoutService._setInitialLayoutMode(this.schema.mode);\r\n    }\r\n    if (this.schema.contentType) {\r\n      this.layoutService._setInitialContentType(this.schema.contentType);\r\n    }\r\n    if (this.schema.fab) {\r\n      this.layoutService._setDefaultFabConfig(this.schema.fab);\r\n    }\r\n    if (this.schema.loader?.showOnInit) {\r\n      this.layoutService.showLoader();\r\n    }\r\n\r\n    this.breadcrumbs = this.computeBreadcrumbs(this.currentUrl);\r\n\r\n    this.subs.push(\r\n      this.layoutService.loading$.subscribe((v) => {\r\n        this.loading = v;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.sidebarOpen$.subscribe((v) => {\r\n        this.sidebarOpen = v;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.sidebarCollapsed$.subscribe((v) => {\r\n        this.sidebarCollapsed = v;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.fabConfig$.subscribe((config) => {\r\n        this.currentFabConfig = config;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.navBadges$.subscribe((m) => {\r\n        this.navBadges = m;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.navVisibility$.subscribe((m) => {\r\n        this.navVisibility = m;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.layoutMode$.subscribe((mode) => {\r\n        this.layoutMode = mode;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.contentType$.subscribe((type) => {\r\n        this.contentType = type;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.layoutService.barVisibility$.subscribe((m) => {\r\n        this.barVisibility = m;\r\n        this.cdr.markForCheck();\r\n      }),\r\n      this.router.events\r\n        .pipe(filter((e): e is NavigationEnd => e instanceof NavigationEnd))\r\n        .subscribe((e) => {\r\n          this.currentUrl = e.urlAfterRedirects;\r\n          this.autoExpandFromUrl(this.currentUrl);\r\n          this.layoutService.closeMobileSidebar();\r\n          this.breadcrumbs = this.computeBreadcrumbs(this.currentUrl);\r\n          this.contentTransitioning = true;\r\n          this.userDropdownOpen = false;\r\n          this.hnavDropdownId = null;\r\n\r\n          if (this.schema.loader?.showOnInit && this.layoutService.isLoading()) {\r\n            const hide = () => this.layoutService.hideLoader();\r\n            const minDuration = this.schema.loader.minDuration ?? 0;\r\n            if (minDuration > 0) {\r\n              setTimeout(hide, minDuration);\r\n            } else {\r\n              hide();\r\n            }\r\n          }\r\n\r\n          this.cdr.markForCheck();\r\n        }),\r\n    );\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subs.forEach((s) => s.unsubscribe());\r\n  }\r\n\r\n  @HostListener('document:keydown.escape')\r\n  onEscape(): void {\r\n    if (this.fabOpen) {\r\n      this.fabOpen = false;\r\n    }\r\n    if (this.userDropdownOpen) {\r\n      this.userDropdownOpen = false;\r\n    }\r\n    if (this.hnavDropdownId) {\r\n      this.hnavDropdownId = null;\r\n    }\r\n    if (this.sidebarOpen) {\r\n      this.layoutService.closeMobileSidebar();\r\n    }\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onDocumentClick(event: MouseEvent): void {\r\n    const target = event.target as HTMLElement;\r\n    if (this.userDropdownOpen && !target.closest('.ui-layout__user-dropdown')) {\r\n      this.userDropdownOpen = false;\r\n      this.cdr.markForCheck();\r\n    }\r\n    if (this.hnavDropdownId && !target.closest('.ui-layout__hnav-dropdown')) {\r\n      this.hnavDropdownId = null;\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  // ── Navigation helpers ──────────────────────────────────────────\r\n\r\n  getItemsForSection(section: UiNavSection): UiNavItem[] {\r\n    const idSet = new Set(section.itemIds);\r\n    return this.schema.navigation.items.filter((i) => idSet.has(i.id));\r\n  }\r\n\r\n  isItemHidden(item: UiNavItem): boolean {\r\n    const override = this.navVisibility.get(item.id);\r\n    if (override !== undefined) return !override;\r\n    return item.hidden ?? false;\r\n  }\r\n\r\n  getItemBadge(item: UiNavItem): string | number | null {\r\n    return this.navBadges.get(item.id) ?? item.badge ?? null;\r\n  }\r\n\r\n  isParentActive(item: UiNavItem): boolean {\r\n    if (!item.route) return false;\r\n    return this.currentUrl === item.route || this.currentUrl.startsWith(item.route + '/');\r\n  }\r\n\r\n  isRouteActive(item: UiNavItem): boolean {\r\n    if (!item.route) return false;\r\n    if (item.routeActiveExact) return this.currentUrl === item.route;\r\n    return this.currentUrl === item.route || this.currentUrl.startsWith(item.route + '/');\r\n  }\r\n\r\n  toggleExpand(item: UiNavItem): void {\r\n    if (this.expandedIds.has(item.id)) {\r\n      this.expandedIds.delete(item.id);\r\n    } else {\r\n      this.expandedIds.add(item.id);\r\n      if (!this.isParentActive(item) && item.children?.length) {\r\n        const firstChild = item.children[0]?.items[0];\r\n        if (firstChild?.route) {\r\n          this.router.navigateByUrl(firstChild.route);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  onNavClick(): void {\r\n    this.layoutService.closeMobileSidebar();\r\n  }\r\n\r\n  // ── FAB ─────────────────────────────────────────────────────────\r\n\r\n  onFabMainClick(): void {\r\n    if (this.currentFabConfig?.secondaryActions?.length) {\r\n      this.fabOpen = !this.fabOpen;\r\n    } else {\r\n      this.currentFabConfig?.mainAction.action?.();\r\n    }\r\n  }\r\n\r\n  onFabAction(action: UiFabAction): void {\r\n    this.fabOpen = false;\r\n    action.action?.();\r\n  }\r\n\r\n  // ── Topbar helpers ──────────────────────────────────────────────\r\n\r\n  getTopbarNavItems(bar: UiBarConfig): UiNavItem[] {\r\n    return bar.navigation?.items ?? this.schema.navigation.items;\r\n  }\r\n\r\n  toggleHnavDropdown(itemId: string): void {\r\n    this.hnavDropdownId = this.hnavDropdownId === itemId ? null : itemId;\r\n  }\r\n\r\n  dismissBar(barId: string): void {\r\n    this.layoutService.hideBar(barId);\r\n  }\r\n\r\n  onDropdownAction(item: { action?: () => void }): void {\r\n    this.userDropdownOpen = false;\r\n    item.action?.();\r\n  }\r\n\r\n  // ── Breadcrumb derivation from nav schema ───────────────────────\r\n\r\n  /**\r\n   * Derives breadcrumbs by matching the current URL against the\r\n   * navigation schema. Walks top-level items, sections, and children\r\n   * to build a trail from root to current page.\r\n   */\r\n  private computeBreadcrumbs(url: string): UiLayoutBreadcrumb[] {\r\n    const items = this.schema.navigation.items;\r\n    const sections = this.schema.navigation.sections;\r\n\r\n    for (const item of items) {\r\n      if (item.type === 'divider' || item.type === 'external') continue;\r\n\r\n      if (item.children?.length) {\r\n        for (const group of item.children) {\r\n          for (const child of group.items) {\r\n            if (child.route && this.urlMatches(url, child.route, child.routeActiveExact)) {\r\n              const trail: UiLayoutBreadcrumb[] = [];\r\n              const sectionLabel = this.findSectionLabel(item.id, sections);\r\n              if (sectionLabel) {\r\n                trail.push({ label: sectionLabel, isLast: false });\r\n              }\r\n              trail.push({ label: item.label, url: item.route, isLast: false });\r\n              trail.push({ label: child.label, isLast: true });\r\n              return trail;\r\n            }\r\n          }\r\n        }\r\n      }\r\n\r\n      if (item.route && this.urlMatches(url, item.route, item.routeActiveExact)) {\r\n        const trail: UiLayoutBreadcrumb[] = [];\r\n        const sectionLabel = this.findSectionLabel(item.id, sections);\r\n        if (sectionLabel) {\r\n          trail.push({ label: sectionLabel, isLast: false });\r\n        }\r\n        trail.push({ label: item.label, isLast: true });\r\n        return trail;\r\n      }\r\n    }\r\n\r\n    return [];\r\n  }\r\n\r\n  private urlMatches(current: string, route: string, exact?: boolean): boolean {\r\n    if (exact) return current === route;\r\n    return current === route || current.startsWith(route + '/');\r\n  }\r\n\r\n  private findSectionLabel(itemId: string, sections?: UiNavSection[]): string | null {\r\n    if (!sections?.length) return null;\r\n    const section = sections.find((s) => s.itemIds.includes(itemId));\r\n    return section?.label ?? null;\r\n  }\r\n\r\n  private autoExpandFromUrl(url: string): void {\r\n    for (const item of this.schema.navigation.items) {\r\n      if (item.children && item.route && (url === item.route || url.startsWith(item.route + '/'))) {\r\n        this.expandedIds.add(item.id);\r\n      }\r\n    }\r\n  }\r\n}\r\n"]}