@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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGF5b3V0LWJ1aWxkZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvbmctdWktc3lzdGVtL3NyYy9saWIvY29tcG9uZW50cy9sYXlvdXQtYnVpbGRlci9sYXlvdXQtYnVpbGRlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FtQkc7QUFDSCxPQUFPLEVBQ0wsU0FBUyxFQUNULEtBQUssRUFHTCx1QkFBdUIsRUFDdkIsaUJBQWlCLEVBQ2pCLGlCQUFpQixFQUNqQixNQUFNLEVBQ04sWUFBWSxHQUNiLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxNQUFNLEVBQUUsWUFBWSxFQUFFLFVBQVUsRUFBRSxnQkFBZ0IsRUFBRSxhQUFhLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUNwRyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUNuRCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQWdCLE1BQU0sRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUU1QyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sa0JBQWtCLENBQUM7OztBQSt0Qm5ELE1BQU0sT0FBTyx3QkFBd0I7SUFsdEJyQztRQXN0Qlcsa0JBQWEsR0FBRyxNQUFNLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDaEMsV0FBTSxHQUFHLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUN4QixRQUFHLEdBQUcsTUFBTSxDQUFDLGlCQUFpQixDQUFDLENBQUM7UUFDekMsU0FBSSxHQUFtQixFQUFFLENBQUM7UUFFbEMsZUFBVSxHQUFHLEVBQUUsQ0FBQztRQUNoQixnQkFBVyxHQUFHLElBQUksR0FBRyxFQUFVLENBQUM7UUFDaEMsWUFBTyxHQUFHLEtBQUssQ0FBQztRQUNoQixxQkFBZ0IsR0FBNkIs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Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0Frc0JUOzhCQUswQixNQUFNO3NCQUFoQyxLQUFLO3VCQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtnQkEySXpCLFFBQVE7c0JBRFAsWUFBWTt1QkFBQyx5QkFBeUI7Z0JBaUJ2QyxlQUFlO3NCQURkLFlBQVk7dUJBQUMsZ0JBQWdCLEVBQUUsQ0FBQyxRQUFRLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogU2NoZW1hLWRyaXZlbiBhcHBsaWNhdGlvbiBzaGVsbCB0aGF0IG1hdGVyaWFsaXplcyBhIGZ1bGwgcmVzcG9uc2l2ZSBsYXlvdXRcclxuICogZnJvbSBhIHNpbmdsZSBgVWlMYXlvdXRTY2hlbWFgIGRlc2NyaXB0b3IuXHJcbiAqXHJcbiAqIEZlYXR1cmVzOlxyXG4gKiAtIFR3byBsYXlvdXQgbW9kZXM6IHNpZGViYXIgKGNvbGxhcHNpYmxlKSBhbmQgdG9wYmFyIChtdWx0aS1iYXIgc3RhY2spXHJcbiAqIC0gUHJvZ3JhbW1hdGljIG1vZGUgc3dpdGNoaW5nIHdpdGggbG9hZGVyIG1hc2tpbmdcclxuICogLSBDb250ZW50IHR5cGU6IGZsdWlkLCBib3hlZCwgZnVsbHNjcmVlblxyXG4gKiAtIFNwZWVkLWRpYWwgRkFCIHdpdGggY29udGV4dC1zZW5zaXRpdmUgYWN0aW9uc1xyXG4gKiAtIE92ZXJsYXkgbG9hZGVyIHdpdGggYm9keS1zY3JvbGwgbG9ja2luZ1xyXG4gKiAtIEF1dG8tZGVyaXZlZCBicmVhZGNydW1icyBmcm9tIG5hdmlnYXRpb24gc2NoZW1hXHJcbiAqIC0gU2tpcC10by1jb250ZW50IGxpbmsgYW5kIGZ1bGwga2V5Ym9hcmQgbmF2aWdhdGlvblxyXG4gKlxyXG4gKiBAc2VsZWN0b3IgdWktbGF5b3V0LWJ1aWxkZXJcclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgaHRtbFxyXG4gKiA8dWktbGF5b3V0LWJ1aWxkZXIgW3NjaGVtYV09XCJsYXlvdXRTY2hlbWFcIiAvPlxyXG4gKiBgYGBcclxuICovXHJcbmltcG9ydCB7XHJcbiAgQ29tcG9uZW50LFxyXG4gIElucHV0LFxyXG4gIE9uSW5pdCxcclxuICBPbkRlc3Ryb3ksXHJcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXHJcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXHJcbiAgVmlld0VuY2Fwc3VsYXRpb24sXHJcbiAgaW5qZWN0LFxyXG4gIEhvc3RMaXN0ZW5lcixcclxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgUm91dGVyLCBSb3V0ZXJPdXRsZXQsIFJvdXRlckxpbmssIFJvdXRlckxpbmtBY3RpdmUsIE5hdmlnYXRpb25FbmQgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xyXG5pbXBvcnQgeyBOZ1RlbXBsYXRlT3V0bGV0IH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgTHVjaWRlQW5ndWxhck1vZHVsZSB9IGZyb20gJ2x1Y2lkZS1hbmd1bGFyJztcclxuaW1wb3J0IHsgU3Vic2NyaXB0aW9uLCBmaWx0ZXIgfSBmcm9tICdyeGpzJztcclxuXHJcbmltcG9ydCB7IFVpTGF5b3V0U2VydmljZSB9IGZyb20gJy4vbGF5b3V0LnNlcnZpY2UnO1xyXG5pbXBvcnQge1xyXG4gIFVpTGF5b3V0U2NoZW1hLFxyXG4gIFVpTGF5b3V0TW9kZSxcclxuICBVaUNvbnRlbnRUeXBlLFxyXG4gIFVpTGF5b3V0QnJlYWRjcnVtYixcclxuICBVaU5hdkl0ZW0sXHJcbiAgVWlOYXZTZWN0aW9uLFxyXG4gIFVpTGF5b3V0RmFiQ29uZmlnLFxyXG4gIFVpRmFiQWN0aW9uLFxyXG4gIFVpQmFyQ29uZmlnLFxyXG59IGZyb20gJy4vbGF5b3V0LWJ1aWxkZXIudHlwZXMnO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICd1aS1sYXlvdXQtYnVpbGRlcicsXHJcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcclxuICBpbXBvcnRzOiBbXHJcbiAgICBSb3V0ZXJPdXRsZXQsXHJcbiAgICBSb3V0ZXJMaW5rLFxyXG4gICAgUm91dGVyTGlua0FjdGl2ZSxcclxuICAgIE5nVGVtcGxhdGVPdXRsZXQsXHJcbiAgICBMdWNpZGVBbmd1bGFyTW9kdWxlLFxyXG4gIF0sXHJcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXHJcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcclxuICBob3N0OiB7IGNsYXNzOiAndWktbGF5b3V0LWJ1aWxkZXItaG9zdCcgfSxcclxuICB0ZW1wbGF0ZTogYFxyXG4gICAgPCEtLSBTa2lwIHRvIGNvbnRlbnQgKGExMXkpIC0tPlxyXG4gICAgPGEgY2xhc3M9XCJ1aS1sYXlvdXRfX3NraXAtbGlua1wiIGhyZWY9XCIjdWktbGF5b3V0LW1haW5cIj5cclxuICAgICAgVmFpIGFsIGNvbnRlbnV0byBwcmluY2lwYWxlXHJcbiAgICA8L2E+XHJcblxyXG4gICAgPGRpdlxyXG4gICAgICBjbGFzcz1cInVpLWxheW91dFwiXHJcbiAgICAgIFtjbGFzcy51aS1sYXlvdXQtLXNpZGViYXItY29sbGFwc2VkXT1cImxheW91dE1vZGUgPT09ICdzaWRlYmFyJyAmJiBzaWRlYmFyQ29sbGFwc2VkXCJcclxuICAgICAgW2NsYXNzLnVpLWxheW91dC0tdG9wYmFyXT1cImxheW91dE1vZGUgPT09ICd0b3BiYXInXCJcclxuICAgID5cclxuICAgICAgPCEtLSDilIDilIDilIAgT3ZlcmxheSBMb2FkZXIg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAIC0tPlxyXG4gICAgICBAaWYgKGxvYWRpbmcpIHtcclxuICAgICAgICA8ZGl2XHJcbiAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fbG9hZGVyLW92ZXJsYXlcIlxyXG4gICAgICAgICAgcm9sZT1cImFsZXJ0XCJcclxuICAgICAgICAgIGFyaWEtbGl2ZT1cImFzc2VydGl2ZVwiXHJcbiAgICAgICAgICBhcmlhLWxhYmVsPVwiQ2FyaWNhbWVudG8gaW4gY29yc29cIlxyXG4gICAgICAgID5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1sYXlvdXRfX2xvYWRlci1zcGlubmVyXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XHJcbiAgICAgICAgICAgIDxzdmcgdmlld0JveD1cIjAgMCA1MCA1MFwiIGNsYXNzPVwidWktbGF5b3V0X19sb2FkZXItc3ZnXCI+XHJcbiAgICAgICAgICAgICAgPGNpcmNsZSBjeD1cIjI1XCIgY3k9XCIyNVwiIHI9XCIyMFwiIGZpbGw9XCJub25lXCIgc3Ryb2tlLXdpZHRoPVwiNFwiIC8+XHJcbiAgICAgICAgICAgIDwvc3ZnPlxyXG4gICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIEBzd2l0Y2ggKGxheW91dE1vZGUpIHtcclxuICAgICAgICA8IS0tIOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkCAtLT5cclxuICAgICAgICA8IS0tIOKVkOKVkOKVkCBTSURFQkFSIE1PREUg4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQIC0tPlxyXG4gICAgICAgIDwhLS0g4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQIC0tPlxyXG4gICAgICAgIEBjYXNlICgnc2lkZWJhcicpIHtcclxuICAgICAgICAgIDwhLS0gTW9iaWxlIFRvcCBCYXIgLS0+XHJcbiAgICAgICAgICA8aGVhZGVyIGNsYXNzPVwidWktbGF5b3V0X190b3BiYXJcIj5cclxuICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19oYW1idXJnZXJcIlxyXG4gICAgICAgICAgICAgIChjbGljayk9XCJsYXlvdXRTZXJ2aWNlLm9wZW5Nb2JpbGVTaWRlYmFyKClcIlxyXG4gICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJBcHJpIG5hdmlnYXppb25lXCJcclxuICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19oYW1idXJnZXItbGluZVwiPjwvc3Bhbj5cclxuICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9faGFtYnVyZ2VyLWxpbmVcIj48L3NwYW4+XHJcbiAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX2hhbWJ1cmdlci1saW5lXCI+PC9zcGFuPlxyXG4gICAgICAgICAgICA8L2J1dHRvbj5cclxuICAgICAgICAgICAgQGlmIChzY2hlbWEuaGVhZGVyLmxvZ28pIHtcclxuICAgICAgICAgICAgICBAaWYgKHNjaGVtYS5oZWFkZXIubG9nby5zcmMpIHtcclxuICAgICAgICAgICAgICAgIDxpbWdcclxuICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX3RvcGJhci1sb2dvLWltZ1wiXHJcbiAgICAgICAgICAgICAgICAgIFtzcmNdPVwic2NoZW1hLmhlYWRlci5sb2dvLnNyY1wiXHJcbiAgICAgICAgICAgICAgICAgIFthbHRdPVwic2NoZW1hLmhlYWRlci5sb2dvLmFsdCB8fCBzY2hlbWEuaGVhZGVyLnRpdGxlXCJcclxuICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgfSBAZWxzZSBpZiAoc2NoZW1hLmhlYWRlci5sb2dvLnRleHQpIHtcclxuICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X190b3BiYXItbG9nb1wiPnt7IHNjaGVtYS5oZWFkZXIubG9nby50ZXh0IH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fdG9wYmFyLXRpdGxlXCI+e3sgc2NoZW1hLmhlYWRlci50aXRsZSB9fTwvc3Bhbj5cclxuICAgICAgICAgIDwvaGVhZGVyPlxyXG5cclxuICAgICAgICAgIDwhLS0gTW9iaWxlIEJhY2tkcm9wIC0tPlxyXG4gICAgICAgICAgQGlmIChzaWRlYmFyT3Blbikge1xyXG4gICAgICAgICAgICA8ZGl2XHJcbiAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2JhY2tkcm9wXCJcclxuICAgICAgICAgICAgICAoY2xpY2spPVwibGF5b3V0U2VydmljZS5jbG9zZU1vYmlsZVNpZGViYXIoKVwiXHJcbiAgICAgICAgICAgICAgYXJpYS1oaWRkZW49XCJ0cnVlXCJcclxuICAgICAgICAgICAgPjwvZGl2PlxyXG4gICAgICAgICAgfVxyXG5cclxuICAgICAgICAgIDwhLS0gQm9keSAoc2lkZWJhciArIGNvbnRlbnQpIC0tPlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9fYm9keVwiPlxyXG4gICAgICAgICAgICA8YXNpZGVcclxuICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fc2lkZWJhclwiXHJcbiAgICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fc2lkZWJhci0tY29sbGFwc2VkXT1cInNpZGViYXJDb2xsYXBzZWRcIlxyXG4gICAgICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX3NpZGViYXItLW9wZW5dPVwic2lkZWJhck9wZW5cIlxyXG4gICAgICAgICAgICAgIHJvbGU9XCJuYXZpZ2F0aW9uXCJcclxuICAgICAgICAgICAgICBhcmlhLWxhYmVsPVwiTmF2aWdhemlvbmUgcHJpbmNpcGFsZVwiXHJcbiAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19zaWRlYmFyLWhlYWRlclwiPlxyXG4gICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9fc2lkZWJhci1icmFuZFwiPlxyXG4gICAgICAgICAgICAgICAgICBAaWYgKHNjaGVtYS5oZWFkZXIubG9nbykge1xyXG4gICAgICAgICAgICAgICAgICAgIEBpZiAoc2NoZW1hLmhlYWRlci5sb2dvLnNyYykge1xyXG4gICAgICAgICAgICAgICAgICAgICAgPGltZ1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fc2lkZWJhci1sb2dvLWltZ1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIFtzcmNdPVwic2NoZW1hLmhlYWRlci5sb2dvLnNyY1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIFthbHRdPVwic2NoZW1hLmhlYWRlci5sb2dvLmFsdCB8fCBzY2hlbWEuaGVhZGVyLnRpdGxlXCJcclxuICAgICAgICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgfSBAZWxzZSBpZiAoc2NoZW1hLmhlYWRlci5sb2dvLnRleHQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19zaWRlYmFyLWxvZ29cIj57eyBzY2hlbWEuaGVhZGVyLmxvZ28udGV4dCB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgQGlmICghc2lkZWJhckNvbGxhcHNlZCB8fCBzaWRlYmFyT3Blbikge1xyXG4gICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19zaWRlYmFyLXRpdGxlXCI+e3sgc2NoZW1hLmhlYWRlci50aXRsZSB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgPC9kaXY+XHJcblxyXG4gICAgICAgICAgICAgICAgQGlmIChzY2hlbWEubmF2aWdhdGlvbi5jb2xsYXBzaWJsZSAhPT0gZmFsc2UpIHtcclxuICAgICAgICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19jb2xsYXBzZS1idG4gdWktbGF5b3V0X19kZXNrdG9wLW9ubHlcIlxyXG4gICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJsYXlvdXRTZXJ2aWNlLnRvZ2dsZVNpZGViYXIoKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgW2F0dHIuYXJpYS1sYWJlbF09XCJzaWRlYmFyQ29sbGFwc2VkID8gJ0VzcGFuZGkgc2lkZWJhcicgOiAnQ29tcHJpbWkgc2lkZWJhcidcIlxyXG4gICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uXHJcbiAgICAgICAgICAgICAgICAgICAgICBuYW1lPVwiY2hldnJvbnMtbGVmdFwiXHJcbiAgICAgICAgICAgICAgICAgICAgICBbc2l6ZV09XCIxNlwiXHJcbiAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fY29sbGFwc2UtaWNvblwiXHJcbiAgICAgICAgICAgICAgICAgICAgICBhcmlhLWhpZGRlbj1cInRydWVcIlxyXG4gICAgICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxyXG4gICAgICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgICAgIDxidXR0b25cclxuICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2Nsb3NlLWJ0biB1aS1sYXlvdXRfX21vYmlsZS1vbmx5XCJcclxuICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cImxheW91dFNlcnZpY2UuY2xvc2VNb2JpbGVTaWRlYmFyKClcIlxyXG4gICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPVwiQ2hpdWRpIG5hdmlnYXppb25lXCJcclxuICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJ4XCIgW3NpemVdPVwiMTZcIiBhcmlhLWhpZGRlbj1cInRydWVcIiAvPlxyXG4gICAgICAgICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgICAgICAgPC9kaXY+XHJcblxyXG4gICAgICAgICAgICAgIEBpZiAoIXNpZGViYXJDb2xsYXBzZWQgfHwgc2lkZWJhck9wZW4pIHtcclxuICAgICAgICAgICAgICAgIDxuYXYgY2xhc3M9XCJ1aS1sYXlvdXRfX3NpZGViYXItbmF2XCI+XHJcbiAgICAgICAgICAgICAgICAgIEBpZiAoc2NoZW1hLm5hdmlnYXRpb24uc2VjdGlvbnM/Lmxlbmd0aCkge1xyXG4gICAgICAgICAgICAgICAgICAgIEBmb3IgKHNlY3Rpb24gb2Ygc2NoZW1hLm5hdmlnYXRpb24uc2VjdGlvbnM7IHRyYWNrIHNlY3Rpb24uaWQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1zZWN0aW9uXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19uYXYtc2VjdGlvbi10aXRsZVwiPnt7IHNlY3Rpb24ubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBmb3IgKGl0ZW0gb2YgZ2V0SXRlbXNGb3JTZWN0aW9uKHNlY3Rpb24pOyB0cmFjayBpdGVtLmlkKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmICghaXNJdGVtSGlkZGVuKGl0ZW0pKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwibmF2SXRlbVRwbDsgY29udGV4dDogeyAkaW1wbGljaXQ6IGl0ZW0gfVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAvPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICB9IEBlbHNlIHtcclxuICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19uYXYtc2VjdGlvblwiPlxyXG4gICAgICAgICAgICAgICAgICAgICAgQGZvciAoaXRlbSBvZiBzY2hlbWEubmF2aWdhdGlvbi5pdGVtczsgdHJhY2sgaXRlbS5pZCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBAaWYgKCFpc0l0ZW1IaWRkZW4oaXRlbSkpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAqbmdUZW1wbGF0ZU91dGxldD1cIm5hdkl0ZW1UcGw7IGNvbnRleHQ6IHsgJGltcGxpY2l0OiBpdGVtIH1cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgPC9uYXY+XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L2FzaWRlPlxyXG5cclxuICAgICAgICAgICAgPCEtLSBNYWluIGNvbnRlbnQgYXJlYSAtLT5cclxuICAgICAgICAgICAgPG1haW5cclxuICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fY29udGVudFwiXHJcbiAgICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fY29udGVudC0tYm94ZWRdPVwiY29udGVudFR5cGUgPT09ICdib3hlZCdcIlxyXG4gICAgICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX2NvbnRlbnQtLWZ1bGxzY3JlZW5dPVwiY29udGVudFR5cGUgPT09ICdmdWxsc2NyZWVuJ1wiXHJcbiAgICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fY29udGVudC0tdHJhbnNpdGlvbmluZ109XCJjb250ZW50VHJhbnNpdGlvbmluZ1wiXHJcbiAgICAgICAgICAgICAgKGFuaW1hdGlvbmVuZCk9XCJjb250ZW50VHJhbnNpdGlvbmluZyA9IGZhbHNlXCJcclxuICAgICAgICAgICAgICBpZD1cInVpLWxheW91dC1tYWluXCJcclxuICAgICAgICAgICAgICB0YWJpbmRleD1cIi0xXCJcclxuICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJwYWdlSGVhZGVyVHBsXCIgLz5cclxuICAgICAgICAgICAgICA8cm91dGVyLW91dGxldCAvPlxyXG4gICAgICAgICAgICA8L21haW4+XHJcbiAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIDwhLS0g4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQIC0tPlxyXG4gICAgICAgIDwhLS0g4pWQ4pWQ4pWQIFRPUEJBUiBNT0RFIOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkCAtLT5cclxuICAgICAgICA8IS0tIOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkOKVkCAtLT5cclxuICAgICAgICBAY2FzZSAoJ3RvcGJhcicpIHtcclxuICAgICAgICAgIDwhLS0gQmFyIHN0YWNrIC0tPlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9fYmFyLXN0YWNrXCI+XHJcbiAgICAgICAgICAgIEBmb3IgKGJhciBvZiB2aXNpYmxlQmFyczsgdHJhY2sgYmFyLmlkKSB7XHJcbiAgICAgICAgICAgICAgQHN3aXRjaCAoYmFyLnR5cGUpIHtcclxuICAgICAgICAgICAgICAgIEBjYXNlICgnbm90aWZpY2F0aW9uJykge1xyXG4gICAgICAgICAgICAgICAgICBAaWYgKGJhci5ub3RpZmljYXRpb24pIHtcclxuICAgICAgICAgICAgICAgICAgICA8ZGl2XHJcbiAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fYmFyIHVpLWxheW91dF9fYmFyLS1ub3RpZmljYXRpb25cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgW2NsYXNzXT1cIid1aS1sYXlvdXRfX2Jhci0tbm90aWZpY2F0aW9uLScgKyAoYmFyLm5vdGlmaWNhdGlvbi52YXJpYW50IHx8ICdwcmltYXJ5JylcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgcm9sZT1cImFsZXJ0XCJcclxuICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICBAaWYgKGJhci5ub3RpZmljYXRpb24uaWNvbikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gW25hbWVdPVwiYmFyLm5vdGlmaWNhdGlvbi5pY29uXCIgW3NpemVdPVwiMTZcIiBhcmlhLWhpZGRlbj1cInRydWVcIiAvPlxyXG4gICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX2Jhci10ZXh0XCI+e3sgYmFyLm5vdGlmaWNhdGlvbi50ZXh0IH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgQGlmIChiYXIubm90aWZpY2F0aW9uLmRpc21pc3NpYmxlICE9PSBmYWxzZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2Jhci1kaXNtaXNzXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAoY2xpY2spPVwiZGlzbWlzc0JhcihiYXIuaWQpXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPVwiQ2hpdWRpIG5vdGlmaWNhXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBuYW1lPVwieFwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICAgICAgICBAY2FzZSAoJ2JyYW5kJykge1xyXG4gICAgICAgICAgICAgICAgICBAaWYgKGJhci5icmFuZDsgYXMgYnJhbmQpIHtcclxuICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19iYXIgdWktbGF5b3V0X19iYXItLWJyYW5kXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19iYXItYnJhbmQtbGVmdFwiPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2hhbWJ1cmdlciB1aS1sYXlvdXRfX3RvcGJhci1oYW1idXJnZXJcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJsYXlvdXRTZXJ2aWNlLm9wZW5Nb2JpbGVTaWRlYmFyKClcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJBcHJpIG5hdmlnYXppb25lXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19oYW1idXJnZXItbGluZVwiPjwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9faGFtYnVyZ2VyLWxpbmVcIj48L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX2hhbWJ1cmdlci1saW5lXCI+PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChicmFuZC5sb2dvOyBhcyBsb2dvKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChsb2dvLnNyYykge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPGltZ1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fYmFyLWxvZ28taW1nXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW3NyY109XCJsb2dvLnNyY1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFthbHRdPVwibG9nby5hbHQgfHwgYnJhbmQudGl0bGUgfHwgJydcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICB9IEBlbHNlIGlmIChsb2dvLnRleHQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19iYXItbG9nb1wiPnt7IGxvZ28udGV4dCB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChicmFuZC50aXRsZTsgYXMgYnJhbmRUaXRsZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19iYXItdGl0bGVcIj57eyBicmFuZFRpdGxlIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgICAgIEBpZiAoYnJhbmQudXNlckRyb3Bkb3duOyBhcyBkcm9wZG93bikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X191c2VyLWRyb3Bkb3duXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX3VzZXItZHJvcGRvd24tdHJpZ2dlclwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoY2xpY2spPVwidXNlckRyb3Bkb3duT3BlbiA9ICF1c2VyRHJvcGRvd25PcGVuXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIFthdHRyLmFyaWEtZXhwYW5kZWRdPVwidXNlckRyb3Bkb3duT3BlblwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWhhc3BvcHVwPVwidHJ1ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChkcm9wZG93bi5hdmF0YXI/LnNyYykge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8aW1nXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX3VzZXItYXZhdGFyXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbc3JjXT1cImRyb3Bkb3duLmF2YXRhciEuc3JjXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbYWx0XT1cImRyb3Bkb3duLmF2YXRhciEuYWx0IHx8IGRyb3Bkb3duLmxhYmVsXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gQGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb25cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbbmFtZV09XCJkcm9wZG93bi5pY29uIHx8ICd1c2VyJ1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW3NpemVdPVwiMThcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFyaWEtaGlkZGVuPVwidHJ1ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fdXNlci1sYWJlbFwiPnt7IGRyb3Bkb3duLmxhYmVsIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJjaGV2cm9uLWRvd25cIiBbc2l6ZV09XCIxNFwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIGNsYXNzPVwidWktbGF5b3V0X191c2VyLWNoZXZyb25cIiAvPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAodXNlckRyb3Bkb3duT3Blbikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9fdXNlci1kcm9wZG93bi1tZW51XCIgcm9sZT1cIm1lbnVcIj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGZvciAoZEl0ZW0gb2YgZHJvcGRvd24uaXRlbXM7IHRyYWNrIGRJdGVtLmlkKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChkSXRlbS5kaXZpZGVyKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8aHIgY2xhc3M9XCJ1aS1sYXlvdXRfX3VzZXItZHJvcGRvd24tZGl2aWRlclwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoZEl0ZW0ucm91dGUpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxhXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X191c2VyLWRyb3Bkb3duLWl0ZW1cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbcm91dGVyTGlua109XCJkSXRlbS5yb3V0ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvbGU9XCJtZW51aXRlbVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJ1c2VyRHJvcGRvd25PcGVuID0gZmFsc2VcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGRJdGVtLmljb24pIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gW25hbWVdPVwiZEl0ZW0uaWNvblwiIFtzaXplXT1cIjE2XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB7eyBkSXRlbS5sYWJlbCB9fVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9hPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gQGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fdXNlci1kcm9wZG93bi1pdGVtXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcm9sZT1cIm1lbnVpdGVtXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cIm9uRHJvcGRvd25BY3Rpb24oZEl0ZW0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChkSXRlbS5pY29uKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cImRJdGVtLmljb25cIiBbc2l6ZV09XCIxNlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAge3sgZEl0ZW0ubGFiZWwgfX1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgICAgIEBjYXNlICgnbmF2aWdhdGlvbicpIHtcclxuICAgICAgICAgICAgICAgICAgPG5hdiBjbGFzcz1cInVpLWxheW91dF9fYmFyIHVpLWxheW91dF9fYmFyLS1uYXZpZ2F0aW9uXCIgcm9sZT1cIm5hdmlnYXRpb25cIiBhcmlhLWxhYmVsPVwiTmF2aWdhemlvbmUgcHJpbmNpcGFsZVwiPlxyXG4gICAgICAgICAgICAgICAgICAgIEBmb3IgKGl0ZW0gb2YgZ2V0VG9wYmFyTmF2SXRlbXMoYmFyKTsgdHJhY2sgaXRlbS5pZCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgQGlmICghaXNJdGVtSGlkZGVuKGl0ZW0pKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoaXRlbS5jaGlsZHJlbj8ubGVuZ3RoKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9faG5hdi1kcm9wZG93blwiIFtjbGFzcy51aS1sYXlvdXRfX2huYXYtZHJvcGRvd24tLW9wZW5dPVwiaG5hdkRyb3Bkb3duSWQgPT09IGl0ZW0uaWRcIj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxidXR0b25cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2huYXYtbGluayB1aS1sYXlvdXRfX2huYXYtbGluay0tcGFyZW50XCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9faG5hdi1saW5rLS1hY3RpdmVdPVwiaXNQYXJlbnRBY3RpdmUoaXRlbSlcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoY2xpY2spPVwidG9nZ2xlSG5hdkRyb3Bkb3duKGl0ZW0uaWQpXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW2F0dHIuYXJpYS1leHBhbmRlZF09XCJobmF2RHJvcGRvd25JZCA9PT0gaXRlbS5pZFwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoaXRlbS5pY29uKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cIml0ZW0uaWNvblwiIFtzaXplXT1cIjE2XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3Bhbj57eyBpdGVtLmxhYmVsIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cImNoZXZyb24tZG93blwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgY2xhc3M9XCJ1aS1sYXlvdXRfX2huYXYtY2hldnJvblwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoaG5hdkRyb3Bkb3duSWQgPT09IGl0ZW0uaWQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWxheW91dF9faG5hdi1kcm9wZG93bi1tZW51XCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGZvciAoZ3JvdXAgb2YgaXRlbS5jaGlsZHJlbjsgdHJhY2sgZ3JvdXAubGFiZWwpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19obmF2LWRyb3Bkb3duLWxhYmVsXCI+e3sgZ3JvdXAubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAZm9yIChjaGlsZCBvZiBncm91cC5pdGVtczsgdHJhY2sgY2hpbGQuaWQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmICghaXNJdGVtSGlkZGVuKGNoaWxkKSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxhXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9faG5hdi1kcm9wZG93bi1pdGVtXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtyb3V0ZXJMaW5rXT1cImNoaWxkLnJvdXRlXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvdXRlckxpbmtBY3RpdmU9XCJ1aS1sYXlvdXRfX2huYXYtZHJvcGRvd24taXRlbS0tYWN0aXZlXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJobmF2RHJvcGRvd25JZCA9IG51bGw7IG9uTmF2Q2xpY2soKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAge3sgY2hpbGQubGFiZWwgfX1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9IEBlbHNlIGlmIChpdGVtLnR5cGUgPT09ICdleHRlcm5hbCcpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8YVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2huYXYtbGlua1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbaHJlZl09XCJpdGVtLmhyZWZcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgW3RhcmdldF09XCJpdGVtLnRhcmdldCB8fCAnX2JsYW5rJ1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbYXR0ci5yZWxdPVwiaXRlbS50YXJnZXQgPT09ICdfYmxhbmsnID8gJ25vb3BlbmVyIG5vcmVmZXJyZXInIDogbnVsbFwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmIChpdGVtLmljb24pIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cIml0ZW0uaWNvblwiIFtzaXplXT1cIjE2XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuPnt7IGl0ZW0ubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cImV4dGVybmFsLWxpbmtcIiBbc2l6ZV09XCIxMlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPC9hPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9IEBlbHNlIGlmIChpdGVtLnR5cGUgIT09ICdkaXZpZGVyJykge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxhXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9faG5hdi1saW5rXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtyb3V0ZXJMaW5rXT1cIml0ZW0ucm91dGVcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcm91dGVyTGlua0FjdGl2ZT1cInVpLWxheW91dF9faG5hdi1saW5rLS1hY3RpdmVcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgW3JvdXRlckxpbmtBY3RpdmVPcHRpb25zXT1cInsgZXhhY3Q6IGl0ZW0ucm91dGVBY3RpdmVFeGFjdCB8fCBmYWxzZSB9XCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGl0ZW0uaWNvbikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gW25hbWVdPVwiaXRlbS5pY29uXCIgW3NpemVdPVwiMTZcIiBhcmlhLWhpZGRlbj1cInRydWVcIiAvPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4+e3sgaXRlbS5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoZ2V0SXRlbUJhZGdlKGl0ZW0pKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19iYWRnZVwiPnt7IGdldEl0ZW1CYWRnZShpdGVtKSB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIDwvbmF2PlxyXG4gICAgICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgICAgIEBjYXNlICgnbGlua3MnKSB7XHJcbiAgICAgICAgICAgICAgICAgIEBpZiAoYmFyLmxpbmtzKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgPGRpdlxyXG4gICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2JhciB1aS1sYXlvdXRfX2Jhci0tbGlua3NcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgW3N0eWxlLmp1c3RpZnktY29udGVudF09XCJiYXIubGlua3MuYWxpZ24gPT09ICdzdGFydCcgPyAnZmxleC1zdGFydCcgOiBiYXIubGlua3MuYWxpZ24gPT09ICdjZW50ZXInID8gJ2NlbnRlcicgOiBiYXIubGlua3MuYWxpZ24gPT09ICdzcGFjZS1iZXR3ZWVuJyA/ICdzcGFjZS1iZXR3ZWVuJyA6ICdmbGV4LWVuZCdcIlxyXG4gICAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICAgIEBmb3IgKGxuayBvZiBiYXIubGlua3MuaXRlbXM7IHRyYWNrIGxuay5sYWJlbCB8fCBsbmsuYXJpYUxhYmVsIHx8ICRpbmRleCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGxuay5yb3V0ZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxhIGNsYXNzPVwidWktbGF5b3V0X19iYXItbGlua1wiIFtyb3V0ZXJMaW5rXT1cImxuay5yb3V0ZVwiIFthdHRyLmFyaWEtbGFiZWxdPVwibG5rLmFyaWFMYWJlbCB8fCBudWxsXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGxuay5pY29uKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBbbmFtZV09XCJsbmsuaWNvblwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAobG5rLmxhYmVsKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuPnt7IGxuay5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH0gQGVsc2UgaWYgKGxuay5ocmVmKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPGFcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19iYXItbGlua1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbaHJlZl09XCJsbmsuaHJlZlwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbdGFyZ2V0XT1cImxuay50YXJnZXQgfHwgJ19zZWxmJ1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbYXR0ci5yZWxdPVwibG5rLnRhcmdldCA9PT0gJ19ibGFuaycgPyAnbm9vcGVuZXIgbm9yZWZlcnJlcicgOiBudWxsXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIFthdHRyLmFyaWEtbGFiZWxdPVwibG5rLmFyaWFMYWJlbCB8fCBudWxsXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGxuay5pY29uKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBbbmFtZV09XCJsbmsuaWNvblwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAobG5rLmxhYmVsKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuPnt7IGxuay5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgPC9kaXY+XHJcblxyXG4gICAgICAgICAgPCEtLSBNb2JpbGUgQmFja2Ryb3AgKHRvcGJhciBtb2RlKSAtLT5cclxuICAgICAgICAgIEBpZiAoc2lkZWJhck9wZW4pIHtcclxuICAgICAgICAgICAgPGRpdlxyXG4gICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19iYWNrZHJvcCB1aS1sYXlvdXRfX2JhY2tkcm9wLS10b3BiYXJcIlxyXG4gICAgICAgICAgICAgIChjbGljayk9XCJsYXlvdXRTZXJ2aWNlLmNsb3NlTW9iaWxlU2lkZWJhcigpXCJcclxuICAgICAgICAgICAgICBhcmlhLWhpZGRlbj1cInRydWVcIlxyXG4gICAgICAgICAgICA+PC9kaXY+XHJcbiAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgPCEtLSBNb2JpbGUgZHJhd2VyICh0b3BiYXIgbW9kZSkgLS0+XHJcbiAgICAgICAgICA8YXNpZGVcclxuICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX3NpZGViYXIgdWktbGF5b3V0X19zaWRlYmFyLS10b3BiYXItZHJhd2VyXCJcclxuICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fc2lkZWJhci0tb3Blbl09XCJzaWRlYmFyT3BlblwiXHJcbiAgICAgICAgICAgIHJvbGU9XCJuYXZpZ2F0aW9uXCJcclxuICAgICAgICAgICAgYXJpYS1sYWJlbD1cIk5hdmlnYXppb25lIHByaW5jaXBhbGVcIlxyXG4gICAgICAgICAgPlxyXG4gICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19zaWRlYmFyLWhlYWRlclwiPlxyXG4gICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1sYXlvdXRfX3NpZGViYXItYnJhbmRcIj5cclxuICAgICAgICAgICAgICAgIEBpZiAoc2NoZW1hLmhlYWRlci5sb2dvKSB7XHJcbiAgICAgICAgICAgICAgICAgIEBpZiAoc2NoZW1hLmhlYWRlci5sb2dvLnNyYykge1xyXG4gICAgICAgICAgICAgICAgICAgIDxpbWcgY2xhc3M9XCJ1aS1sYXlvdXRfX3NpZGViYXItbG9nby1pbWdcIiBbc3JjXT1cInNjaGVtYS5oZWFkZXIubG9nby5zcmNcIiBbYWx0XT1cInNjaGVtYS5oZWFkZXIubG9nby5hbHQgfHwgc2NoZW1hLmhlYWRlci50aXRsZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgIH0gQGVsc2UgaWYgKHNjaGVtYS5oZWFkZXIubG9nby50ZXh0KSB7XHJcbiAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX3NpZGViYXItbG9nb1wiPnt7IHNjaGVtYS5oZWFkZXIubG9nby50ZXh0IH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fc2lkZWJhci10aXRsZVwiPnt7IHNjaGVtYS5oZWFkZXIudGl0bGUgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgPGJ1dHRvbiBjbGFzcz1cInVpLWxheW91dF9fY2xvc2UtYnRuXCIgKGNsaWNrKT1cImxheW91dFNlcnZpY2UuY2xvc2VNb2JpbGVTaWRlYmFyKClcIiBhcmlhLWxhYmVsPVwiQ2hpdWRpIG5hdmlnYXppb25lXCI+XHJcbiAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cInhcIiBbc2l6ZV09XCIxNlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICA8bmF2IGNsYXNzPVwidWktbGF5b3V0X19zaWRlYmFyLW5hdlwiPlxyXG4gICAgICAgICAgICAgIEBpZiAoc2NoZW1hLm5hdmlnYXRpb24uc2VjdGlvbnM/Lmxlbmd0aCkge1xyXG4gICAgICAgICAgICAgICAgQGZvciAoc2VjdGlvbiBvZiBzY2hlbWEubmF2aWdhdGlvbi5zZWN0aW9uczsgdHJhY2sgc2VjdGlvbi5pZCkge1xyXG4gICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19uYXYtc2VjdGlvblwiPlxyXG4gICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19uYXYtc2VjdGlvbi10aXRsZVwiPnt7IHNlY3Rpb24ubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgQGZvciAoaXRlbSBvZiBnZXRJdGVtc0ZvclNlY3Rpb24oc2VjdGlvbik7IHRyYWNrIGl0ZW0uaWQpIHtcclxuICAgICAgICAgICAgICAgICAgICAgIEBpZiAoIWlzSXRlbUhpZGRlbihpdGVtKSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ1RlbXBsYXRlT3V0bGV0PVwibmF2SXRlbVRwbDsgY29udGV4dDogeyAkaW1wbGljaXQ6IGl0ZW0gfVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICB9IEBlbHNlIHtcclxuICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1zZWN0aW9uXCI+XHJcbiAgICAgICAgICAgICAgICAgIEBmb3IgKGl0ZW0gb2Ygc2NoZW1hLm5hdmlnYXRpb24uaXRlbXM7IHRyYWNrIGl0ZW0uaWQpIHtcclxuICAgICAgICAgICAgICAgICAgICBAaWYgKCFpc0l0ZW1IaWRkZW4oaXRlbSkpIHtcclxuICAgICAgICAgICAgICAgICAgICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJuYXZJdGVtVHBsOyBjb250ZXh0OiB7ICRpbXBsaWNpdDogaXRlbSB9XCIgLz5cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgPC9uYXY+XHJcbiAgICAgICAgICA8L2FzaWRlPlxyXG5cclxuICAgICAgICAgIDwhLS0gTWFpbiBjb250ZW50IGFyZWEgKHRvcGJhciBtb2RlKSAtLT5cclxuICAgICAgICAgIDxtYWluXHJcbiAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19jb250ZW50IHVpLWxheW91dF9fY29udGVudC0tdG9wYmFyXCJcclxuICAgICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fY29udGVudC0tYm94ZWRdPVwiY29udGVudFR5cGUgPT09ICdib3hlZCdcIlxyXG4gICAgICAgICAgICBbY2xhc3MudWktbGF5b3V0X19jb250ZW50LS1mdWxsc2NyZWVuXT1cImNvbnRlbnRUeXBlID09PSAnZnVsbHNjcmVlbidcIlxyXG4gICAgICAgICAgICBbY2xhc3MudWktbGF5b3V0X19jb250ZW50LS10cmFuc2l0aW9uaW5nXT1cImNvbnRlbnRUcmFuc2l0aW9uaW5nXCJcclxuICAgICAgICAgICAgKGFuaW1hdGlvbmVuZCk9XCJjb250ZW50VHJhbnNpdGlvbmluZyA9IGZhbHNlXCJcclxuICAgICAgICAgICAgaWQ9XCJ1aS1sYXlvdXQtbWFpblwiXHJcbiAgICAgICAgICAgIHRhYmluZGV4PVwiLTFcIlxyXG4gICAgICAgICAgPlxyXG4gICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ1RlbXBsYXRlT3V0bGV0PVwicGFnZUhlYWRlclRwbFwiIC8+XHJcbiAgICAgICAgICAgIDxyb3V0ZXItb3V0bGV0IC8+XHJcbiAgICAgICAgICA8L21haW4+XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcblxyXG4gICAgICA8IS0tIOKUgOKUgOKUgCBGb290ZXIg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAIC0tPlxyXG4gICAgICBAaWYgKHNjaGVtYS5mb290ZXIpIHtcclxuICAgICAgICA8Zm9vdGVyIGNsYXNzPVwidWktbGF5b3V0X19mb290ZXJcIj5cclxuICAgICAgICAgIEBpZiAoc2NoZW1hLmZvb3Rlci5saW5rcz8ubGVuZ3RoKSB7XHJcbiAgICAgICAgICAgIDxuYXYgY2xhc3M9XCJ1aS1sYXlvdXRfX2Zvb3Rlci1saW5rc1wiIGFyaWEtbGFiZWw9XCJMaW5rIGZvb3RlclwiPlxyXG4gICAgICAgICAgICAgIEBmb3IgKGxpbmsgb2Ygc2NoZW1hLmZvb3Rlci5saW5rczsgdHJhY2sgbGluay5sYWJlbCkge1xyXG4gICAgICAgICAgICAgICAgQGlmIChsaW5rLnJvdXRlKSB7XHJcbiAgICAgICAgICAgICAgICAgIDxhIGNsYXNzPVwidWktbGF5b3V0X19mb290ZXItbGlua1wiIFtyb3V0ZXJMaW5rXT1cImxpbmsucm91dGVcIj5cclxuICAgICAgICAgICAgICAgICAgICBAaWYgKGxpbmsuaWNvbikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cImxpbmsuaWNvblwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAge3sgbGluay5sYWJlbCB9fVxyXG4gICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICB9IEBlbHNlIGlmIChsaW5rLmhyZWYpIHtcclxuICAgICAgICAgICAgICAgICAgPGFcclxuICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fZm9vdGVyLWxpbmtcIlxyXG4gICAgICAgICAgICAgICAgICAgIFtocmVmXT1cImxpbmsuaHJlZlwiXHJcbiAgICAgICAgICAgICAgICAgICAgW3RhcmdldF09XCJsaW5rLnRhcmdldCB8fCAnX3NlbGYnXCJcclxuICAgICAgICAgICAgICAgICAgICBbYXR0ci5yZWxdPVwibGluay50YXJnZXQgPT09ICdfYmxhbmsnID8gJ25vb3BlbmVyIG5vcmVmZXJyZXInIDogbnVsbFwiXHJcbiAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICBAaWYgKGxpbmsuaWNvbikge1xyXG4gICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cImxpbmsuaWNvblwiIFtzaXplXT1cIjE0XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAge3sgbGluay5sYWJlbCB9fVxyXG4gICAgICAgICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L25hdj5cclxuICAgICAgICAgIH1cclxuICAgICAgICAgIEBpZiAoc2NoZW1hLmZvb3Rlci50ZXh0KSB7XHJcbiAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19mb290ZXItdGV4dFwiPnt7IHNjaGVtYS5mb290ZXIudGV4dCB9fTwvc3Bhbj5cclxuICAgICAgICAgIH1cclxuICAgICAgICA8L2Zvb3Rlcj5cclxuICAgICAgfVxyXG5cclxuICAgICAgPCEtLSDilIDilIDilIAgU3BlZWQgRGlhbCBGQUIg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAIC0tPlxyXG4gICAgICBAaWYgKGN1cnJlbnRGYWJDb25maWcpIHtcclxuICAgICAgICA8ZGl2XHJcbiAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fZmFiXCJcclxuICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX2ZhYi0tb3Blbl09XCJmYWJPcGVuXCJcclxuICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX2ZhYi0tYm90dG9tLWxlZnRdPVwiY3VycmVudEZhYkNvbmZpZy5wb3NpdGlvbiA9PT0gJ2JvdHRvbS1sZWZ0J1wiXHJcbiAgICAgICAgICByb2xlPVwiY29tcGxlbWVudGFyeVwiXHJcbiAgICAgICAgICBhcmlhLWxhYmVsPVwiQXppb25pIHJhcGlkZVwiXHJcbiAgICAgICAgPlxyXG4gICAgICAgICAgQGlmIChmYWJPcGVuICYmIGN1cnJlbnRGYWJDb25maWcuc2Vjb25kYXJ5QWN0aW9ucz8ubGVuZ3RoKSB7XHJcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1sYXlvdXRfX2ZhYi1hY3Rpb25zXCI+XHJcbiAgICAgICAgICAgICAgQGZvciAoYWN0aW9uIG9mIGN1cnJlbnRGYWJDb25maWcuc2Vjb25kYXJ5QWN0aW9uczsgdHJhY2sgYWN0aW9uLmlkOyBsZXQgaSA9ICRpbmRleCkge1xyXG4gICAgICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fZmFiLWFjdGlvblwiXHJcbiAgICAgICAgICAgICAgICAgIFtjbGFzc109XCIndWktbGF5b3V0X19mYWItYWN0aW9uLS0nICsgKGFjdGlvbi52YXJpYW50IHx8ICduZXV0cmFsJylcIlxyXG4gICAgICAgICAgICAgICAgICBbYXR0ci5hcmlhLWxhYmVsXT1cImFjdGlvbi5hcmlhTGFiZWwgfHwgYWN0aW9uLmxhYmVsXCJcclxuICAgICAgICAgICAgICAgICAgW3RpdGxlXT1cImFjdGlvbi5sYWJlbFwiXHJcbiAgICAgICAgICAgICAgICAgIFtzdHlsZS5hbmltYXRpb24tZGVsYXldPVwiKGkgKiA1MCkgKyAnbXMnXCJcclxuICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cIm9uRmFiQWN0aW9uKGFjdGlvbilcIlxyXG4gICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gW25hbWVdPVwiYWN0aW9uLmljb25cIiBbc2l6ZV09XCIxOFwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19mYWItYWN0aW9uLWxhYmVsXCI+e3sgYWN0aW9uLmxhYmVsIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgIH1cclxuICAgICAgICAgIDxidXR0b25cclxuICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2ZhYi1tYWluXCJcclxuICAgICAgICAgICAgW2NsYXNzXT1cIid1aS1sYXlvdXRfX2ZhYi1tYWluLS0nICsgKGN1cnJlbnRGYWJDb25maWcubWFpbkFjdGlvbi52YXJpYW50IHx8ICdwcmltYXJ5JylcIlxyXG4gICAgICAgICAgICBbYXR0ci5hcmlhLWxhYmVsXT1cImN1cnJlbnRGYWJDb25maWcubWFpbkFjdGlvbi5hcmlhTGFiZWwgfHwgY3VycmVudEZhYkNvbmZpZy5tYWluQWN0aW9uLmxhYmVsXCJcclxuICAgICAgICAgICAgW2F0dHIuYXJpYS1leHBhbmRlZF09XCJjdXJyZW50RmFiQ29uZmlnLnNlY29uZGFyeUFjdGlvbnM/Lmxlbmd0aCA/IGZhYk9wZW4gOiBudWxsXCJcclxuICAgICAgICAgICAgKGNsaWNrKT1cIm9uRmFiTWFpbkNsaWNrKClcIlxyXG4gICAgICAgICAgPlxyXG4gICAgICAgICAgICA8bHVjaWRlLWljb25cclxuICAgICAgICAgICAgICBbbmFtZV09XCJjdXJyZW50RmFiQ29uZmlnLm1haW5BY3Rpb24uaWNvblwiXHJcbiAgICAgICAgICAgICAgW3NpemVdPVwiMjRcIlxyXG4gICAgICAgICAgICAgIGFyaWEtaGlkZGVuPVwidHJ1ZVwiXHJcbiAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2ZhYi1tYWluLWljb25cIlxyXG4gICAgICAgICAgICAvPlxyXG4gICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIDwhLS0g4pSA4pSA4pSAIEJvdHRvbSBOYXZpZ2F0aW9uIChtb2JpbGUsIHNpZGViYXIgbW9kZSkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAIC0tPlxyXG4gICAgICBAaWYgKGxheW91dE1vZGUgPT09ICdzaWRlYmFyJyAmJiBib3R0b21OYXZJdGVtcy5sZW5ndGggPiAwICYmIG1vYmlsZU1vZGUgIT09ICdkcmF3ZXInKSB7XHJcbiAgICAgICAgPG5hdlxyXG4gICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2JvdHRvbS1uYXZcIlxyXG4gICAgICAgICAgcm9sZT1cIm5hdmlnYXRpb25cIlxyXG4gICAgICAgICAgYXJpYS1sYWJlbD1cIk5hdmlnYXppb25lIHJhcGlkYVwiXHJcbiAgICAgICAgPlxyXG4gICAgICAgICAgQGZvciAoaXRlbSBvZiBib3R0b21OYXZJdGVtczsgdHJhY2sgaXRlbS5pZCkge1xyXG4gICAgICAgICAgICA8YVxyXG4gICAgICAgICAgICAgIGNsYXNzPVwidWktbGF5b3V0X19ib3R0b20tbmF2LWl0ZW1cIlxyXG4gICAgICAgICAgICAgIFtyb3V0ZXJMaW5rXT1cIml0ZW0ucm91dGVcIlxyXG4gICAgICAgICAgICAgIHJvdXRlckxpbmtBY3RpdmU9XCJ1aS1sYXlvdXRfX2JvdHRvbS1uYXYtaXRlbS0tYWN0aXZlXCJcclxuICAgICAgICAgICAgICBbcm91dGVyTGlua0FjdGl2ZU9wdGlvbnNdPVwieyBleGFjdDogaXRlbS5yb3V0ZUFjdGl2ZUV4YWN0IHx8IGZhbHNlIH1cIlxyXG4gICAgICAgICAgICAgIFthdHRyLmFyaWEtY3VycmVudF09XCJpc1JvdXRlQWN0aXZlKGl0ZW0pID8gJ3BhZ2UnIDogbnVsbFwiXHJcbiAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICBAaWYgKGl0ZW0uaWNvbikge1xyXG4gICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIFtuYW1lXT1cIml0ZW0uaWNvblwiIFtzaXplXT1cIjIwXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX2JvdHRvbS1uYXYtbGFiZWxcIj57eyBpdGVtLmxhYmVsIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgIEBpZiAoZ2V0SXRlbUJhZGdlKGl0ZW0pKSB7XHJcbiAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fYmFkZ2VcIiBhcmlhLWxhYmVsPVwiTm90aWZpY2FcIj57eyBnZXRJdGVtQmFkZ2UoaXRlbSkgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L2E+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgPC9uYXY+XHJcbiAgICAgIH1cclxuICAgIDwvZGl2PlxyXG5cclxuICAgIDwhLS0g4pWQ4pWQ4pWQIFNIQVJFRCBURU1QTEFURVMg4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQIC0tPlxyXG5cclxuICAgIDwhLS0gUGFnZSBoZWFkZXIgd2l0aCBhdXRvLWRlcml2ZWQgYnJlYWRjcnVtYnMgLS0+XHJcbiAgICA8bmctdGVtcGxhdGUgI3BhZ2VIZWFkZXJUcGw+XHJcbiAgICAgIEBpZiAoc2NoZW1hLnBhZ2VIZWFkZXI/LnNob3cgIT09IGZhbHNlICYmIGJyZWFkY3J1bWJzLmxlbmd0aCA+IDApIHtcclxuICAgICAgICA8aGVhZGVyIGNsYXNzPVwidWktbGF5b3V0X19wYWdlLWhlYWRlclwiPlxyXG4gICAgICAgICAgPG5hdiBhcmlhLWxhYmVsPVwiQnJlYWRjcnVtYlwiIGNsYXNzPVwidWktbGF5b3V0X19icmVhZGNydW1iLW5hdlwiPlxyXG4gICAgICAgICAgICA8b2wgY2xhc3M9XCJ1aS1sYXlvdXRfX2JyZWFkY3J1bWJcIiBpdGVtc2NvcGUgaXRlbXR5cGU9XCJodHRwczovL3NjaGVtYS5vcmcvQnJlYWRjcnVtYkxpc3RcIj5cclxuICAgICAgICAgICAgICBAaWYgKHNjaGVtYS5wYWdlSGVhZGVyPy5zaG93SG9tZSAhPT0gZmFsc2UpIHtcclxuICAgICAgICAgICAgICAgIDxsaSBjbGFzcz1cInVpLWxheW91dF9fYnJlYWRjcnVtYi1pdGVtXCIgaXRlbXByb3A9XCJpdGVtTGlzdEVsZW1lbnRcIiBpdGVtc2NvcGUgaXRlbXR5cGU9XCJodHRwczovL3NjaGVtYS5vcmcvTGlzdEl0ZW1cIj5cclxuICAgICAgICAgICAgICAgICAgPGEgaXRlbXByb3A9XCJpdGVtXCIgW3JvdXRlckxpbmtdPVwic2NoZW1hLnBhZ2VIZWFkZXI/LmhvbWVSb3V0ZSB8fCAnLydcIiBjbGFzcz1cInVpLWxheW91dF9fYnJlYWRjcnVtYi1saW5rXCIgYXJpYS1sYWJlbD1cIkhvbWVcIj5cclxuICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cImhvbWVcIiBbc2l6ZV09XCIxNFwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgPHNwYW4gaXRlbXByb3A9XCJuYW1lXCIgY2xhc3M9XCJ1aS1sYXlvdXRfX2JyZWFkY3J1bWItdGV4dFwiPkhvbWU8L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgIDwvYT5cclxuICAgICAgICAgICAgICAgICAgPG1ldGEgaXRlbXByb3A9XCJwb3NpdGlvblwiIGNvbnRlbnQ9XCIxXCIgLz5cclxuICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX2JyZWFkY3J1bWItc2VwXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJjaGV2cm9uLXJpZ2h0XCIgW3NpemVdPVwiMTRcIiAvPlxyXG4gICAgICAgICAgICAgICAgICA8L3NwYW4+XHJcbiAgICAgICAgICAgICAgICA8L2xpPlxyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICBAZm9yIChjcnVtYiBvZiBicmVhZGNydW1iczsgdHJhY2sgY3J1bWIubGFiZWw7IGxldCBpID0gJGluZGV4KSB7XHJcbiAgICAgICAgICAgICAgICA8bGlcclxuICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX2JyZWFkY3J1bWItaXRlbVwiXHJcbiAgICAgICAgICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX2JyZWFkY3J1bWItaXRlbS0tY3VycmVudF09XCJjcnVtYi5pc0xhc3RcIlxyXG4gICAgICAgICAgICAgICAgICBpdGVtcHJvcD1cIml0ZW1MaXN0RWxlbWVudFwiXHJcbiAgICAgICAgICAgICAgICAgIGl0ZW1zY29wZVxyXG4gICAgICAgICAgICAgICAgICBpdGVtdHlwZT1cImh0dHBzOi8vc2NoZW1hLm9yZy9MaXN0SXRlbVwiXHJcbiAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgIEBpZiAoIWNydW1iLmlzTGFzdCAmJiBjcnVtYi51cmwpIHtcclxuICAgICAgICAgICAgICAgICAgICA8YSBpdGVtcHJvcD1cIml0ZW1cIiBbcm91dGVyTGlua109XCJjcnVtYi51cmxcIiBjbGFzcz1cInVpLWxheW91dF9fYnJlYWRjcnVtYi1saW5rXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBpdGVtcHJvcD1cIm5hbWVcIiBjbGFzcz1cInVpLWxheW91dF9fYnJlYWRjcnVtYi10ZXh0XCI+e3sgY3J1bWIubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgPC9hPlxyXG4gICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19icmVhZGNydW1iLXNlcFwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPlxyXG4gICAgICAgICAgICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJjaGV2cm9uLXJpZ2h0XCIgW3NpemVdPVwiMTRcIiAvPlxyXG4gICAgICAgICAgICAgICAgICAgIDwvc3Bhbj5cclxuICAgICAgICAgICAgICAgICAgfSBAZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgPHNwYW4gaXRlbXByb3A9XCJuYW1lXCIgY2xhc3M9XCJ1aS1sYXlvdXRfX2JyZWFkY3J1bWItdGV4dCB1aS1sYXlvdXRfX2JyZWFkY3J1bWItdGV4dC0tY3VycmVudFwiIGFyaWEtY3VycmVudD1cInBhZ2VcIj5cclxuICAgICAgICAgICAgICAgICAgICAgIHt7IGNydW1iLmxhYmVsIH19XHJcbiAgICAgICAgICAgICAgICAgICAgPC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIDxtZXRhIGl0ZW1wcm9wPVwicG9zaXRpb25cIiBbYXR0ci5jb250ZW50XT1cIihzY2hlbWEucGFnZUhlYWRlcj8uc2hvd0hvbWUgIT09IGZhbHNlKSA/IGkgKyAyIDogaSArIDFcIiAvPlxyXG4gICAgICAgICAgICAgICAgPC9saT5cclxuICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIDwvb2w+XHJcbiAgICAgICAgICA8L25hdj5cclxuICAgICAgICAgIDxoMSBjbGFzcz1cInVpLWxheW91dF9fcGFnZS10aXRsZVwiPnt7IHBhZ2VUaXRsZSB9fTwvaDE+XHJcbiAgICAgICAgPC9oZWFkZXI+XHJcbiAgICAgIH1cclxuICAgIDwvbmctdGVtcGxhdGU+XHJcblxyXG4gICAgPCEtLSBOYXYgaXRlbSB0ZW1wbGF0ZSAoc2lkZWJhciwgcmVjdXJzaXZlLWZyaWVuZGx5KSAtLT5cclxuICAgIDxuZy10ZW1wbGF0ZSAjbmF2SXRlbVRwbCBsZXQtaXRlbT5cclxuICAgICAgQGlmIChpdGVtLnR5cGUgPT09ICdkaXZpZGVyJykge1xyXG4gICAgICAgIDxociBjbGFzcz1cInVpLWxheW91dF9fbmF2LWRpdmlkZXJcIiAvPlxyXG4gICAgICB9IEBlbHNlIGlmIChpdGVtLnR5cGUgPT09ICdleHRlcm5hbCcpIHtcclxuICAgICAgICA8YVxyXG4gICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1saW5rXCJcclxuICAgICAgICAgIFtocmVmXT1cIml0ZW0uaHJlZlwiXHJcbiAgICAgICAgICBbdGFyZ2V0XT1cIml0ZW0udGFyZ2V0IHx8ICdfYmxhbmsnXCJcclxuICAgICAgICAgIFthdHRyLnJlbF09XCJpdGVtLnRhcmdldCA9PT0gJ19ibGFuaycgPyAnbm9vcGVuZXIgbm9yZWZlcnJlcicgOiBudWxsXCJcclxuICAgICAgICAgIChjbGljayk9XCJvbk5hdkNsaWNrKClcIlxyXG4gICAgICAgID5cclxuICAgICAgICAgIEBpZiAoaXRlbS5pY29uKSB7XHJcbiAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBbbmFtZV09XCJpdGVtLmljb25cIiBbc2l6ZV09XCIxNlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIGNsYXNzPVwidWktbGF5b3V0X19uYXYtaWNvblwiIC8+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWxhYmVsXCI+e3sgaXRlbS5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgICAgIDxsdWNpZGUtaWNvbiBuYW1lPVwiZXh0ZXJuYWwtbGlua1wiIFtzaXplXT1cIjEyXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1leHRlcm5hbFwiIC8+XHJcbiAgICAgICAgICBAaWYgKGdldEl0ZW1CYWRnZShpdGVtKSkge1xyXG4gICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fYmFkZ2VcIj57eyBnZXRJdGVtQmFkZ2UoaXRlbSkgfX08L3NwYW4+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgPC9hPlxyXG4gICAgICB9IEBlbHNlIGlmIChpdGVtLmNoaWxkcmVuPy5sZW5ndGgpIHtcclxuICAgICAgICA8YnV0dG9uXHJcbiAgICAgICAgICBjbGFzcz1cInVpLWxheW91dF9fbmF2LWxpbmsgdWktbGF5b3V0X19uYXYtbGluay0tcGFyZW50XCJcclxuICAgICAgICAgIFtjbGFzcy51aS1sYXlvdXRfX25hdi1saW5rLS1hY3RpdmVdPVwiaXNQYXJlbnRBY3RpdmUoaXRlbSlcIlxyXG4gICAgICAgICAgW2NsYXNzLnVpLWxheW91dF9fbmF2LWxpbmstLWV4cGFuZGVkXT1cImV4cGFuZGVkSWRzLmhhcyhpdGVtLmlkKVwiXHJcbiAgICAgICAgICBbYXR0ci5hcmlhLWV4cGFuZGVkXT1cImV4cGFuZGVkSWRzLmhhcyhpdGVtLmlkKVwiXHJcbiAgICAgICAgICAoY2xpY2spPVwidG9nZ2xlRXhwYW5kKGl0ZW0pXCJcclxuICAgICAgICA+XHJcbiAgICAgICAgICBAaWYgKGl0ZW0uaWNvbikge1xyXG4gICAgICAgICAgICA8bHVjaWRlLWljb24gW25hbWVdPVwiaXRlbS5pY29uXCIgW3NpemVdPVwiMTZcIiBhcmlhLWhpZGRlbj1cInRydWVcIiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWljb25cIiAvPlxyXG4gICAgICAgICAgfVxyXG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1sYWJlbFwiPnt7IGl0ZW0ubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICBAaWYgKGdldEl0ZW1CYWRnZShpdGVtKSkge1xyXG4gICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fYmFkZ2VcIj57eyBnZXRJdGVtQmFkZ2UoaXRlbSkgfX08L3NwYW4+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWNoZXZyb25cIiBhcmlhLWhpZGRlbj1cInRydWVcIj5cclxuICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJjaGV2cm9uLXJpZ2h0XCIgW3NpemVdPVwiMTRcIiAvPlxyXG4gICAgICAgICAgPC9zcGFuPlxyXG4gICAgICAgIDwvYnV0dG9uPlxyXG4gICAgICAgIEBpZiAoZXhwYW5kZWRJZHMuaGFzKGl0ZW0uaWQpKSB7XHJcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktbGF5b3V0X19uYXYtY2hpbGRyZW5cIj5cclxuICAgICAgICAgICAgQGZvciAoZ3JvdXAgb2YgaXRlbS5jaGlsZHJlbjsgdHJhY2sgZ3JvdXAubGFiZWwpIHtcclxuICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWdyb3VwLWxhYmVsXCI+e3sgZ3JvdXAubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgQGZvciAoY2hpbGQgb2YgZ3JvdXAuaXRlbXM7IHRyYWNrIGNoaWxkLmlkKSB7XHJcbiAgICAgICAgICAgICAgICBAaWYgKCFpc0l0ZW1IaWRkZW4oY2hpbGQpKSB7XHJcbiAgICAgICAgICAgICAgICAgIDxhXHJcbiAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1saW5rIHVpLWxheW91dF9fbmF2LWxpbmstLWNoaWxkXCJcclxuICAgICAgICAgICAgICAgICAgICBbcm91dGVyTGlua109XCJjaGlsZC5yb3V0ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgcm91dGVyTGlua0FjdGl2ZT1cInVpLWxheW91dF9fbmF2LWxpbmstLWFjdGl2ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cIm9uTmF2Q2xpY2soKVwiXHJcbiAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWxhYmVsXCI+e3sgY2hpbGQubGFiZWwgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgQGlmIChnZXRJdGVtQmFkZ2UoY2hpbGQpKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fYmFkZ2VcIj57eyBnZXRJdGVtQmFkZ2UoY2hpbGQpIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgPC9hPlxyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgfVxyXG4gICAgICB9IEBlbHNlIHtcclxuICAgICAgICA8YVxyXG4gICAgICAgICAgY2xhc3M9XCJ1aS1sYXlvdXRfX25hdi1saW5rXCJcclxuICAgICAgICAgIFtyb3V0ZXJMaW5rXT1cIml0ZW0ucm91dGVcIlxyXG4gICAgICAgICAgcm91dGVyTGlua0FjdGl2ZT1cInVpLWxheW91dF9fbmF2LWxpbmstLWFjdGl2ZVwiXHJcbiAgICAgICAgICBbcm91dGVyTGlua0FjdGl2ZU9wdGlvbnNdPVwieyBleGFjdDogaXRlbS5yb3V0ZUFjdGl2ZUV4YWN0IHx8IGZhbHNlIH1cIlxyXG4gICAgICAgICAgW2F0dHIuYXJpYS1jdXJyZW50XT1cImlzUm91dGVBY3RpdmUoaXRlbSkgPyAncGFnZScgOiBudWxsXCJcclxuICAgICAgICAgIChjbGljayk9XCJvbk5hdkNsaWNrKClcIlxyXG4gICAgICAgID5cclxuICAgICAgICAgIEBpZiAoaXRlbS5pY29uKSB7XHJcbiAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBbbmFtZV09XCJpdGVtLmljb25cIiBbc2l6ZV09XCIxNlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIGNsYXNzPVwidWktbGF5b3V0X19uYXYtaWNvblwiIC8+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWxheW91dF9fbmF2LWxhYmVsXCI+e3sgaXRlbS5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgICAgIEBpZiAoZ2V0SXRlbUJhZGdlKGl0ZW0pKSB7XHJcbiAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktbGF5b3V0X19iYWRnZVwiPnt7IGdldEl0ZW1CYWRnZShpdGVtKSB9fTwvc3Bhbj5cclxuICAgICAgICAgIH1cclxuICAgICAgICA8L2E+XHJcbiAgICAgIH1cclxuICAgIDwvbmctdGVtcGxhdGU+XHJcbiAgYCxcclxuICBzdHlsZVVybDogJy4vbGF5b3V0LWJ1aWxkZXIuY29tcG9uZW50LnNjc3MnLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgVWlMYXlvdXRCdWlsZGVyQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0LCBPbkRlc3Ryb3kge1xyXG4gIC8qKiBGdWxsIGxheW91dCBkZXNjcmlwdG9yLiAqL1xyXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pIHNjaGVtYSE6IFVpTGF5b3V0U2NoZW1hO1xyXG5cclxuICByZWFkb25seSBsYXlvdXRTZXJ2aWNlID0gaW5qZWN0KFVpTGF5b3V0U2VydmljZSk7XHJcbiAgcHJpdmF0ZSByZWFkb25seSByb3V0ZXIgPSBpbmplY3QoUm91dGVyKTtcclxuICBwcml2YXRlIHJlYWRvbmx5IGNkciA9IGluamVjdChDaGFuZ2VEZXRlY3RvclJlZik7XHJcbiAgcHJpdmF0ZSBzdWJzOiBTdWJzY3JpcHRpb25bXSA9IFtdO1xyXG5cclxuICBjdXJyZW50VXJsID0gJyc7XHJcbiAgZXhwYW5kZWRJZHMgPSBuZXcgU2V0PHN0cmluZz4oKTtcclxuICBmYWJPcGVuID0gZmFsc2U7XHJcbiAgY3VycmVudEZhYkNvbmZpZzogVWlMYXlvdXRGYWJDb25maWcgfCBudWxsID0gbnVsbDtcclxuXHJcbiAgbG9hZGluZyA9IGZhbHNlO1xyXG4gIHNpZGViYXJPcGVuID0gZmFsc2U7XHJcbiAgc2lkZWJhckNvbGxhcHNlZCA9IGZhbHNlO1xyXG4gIGxheW91dE1vZGU6IFVpTGF5b3V0TW9kZSA9ICdzaWRlYmFyJztcclxuICBjb250ZW50VHlwZTogVWlDb250ZW50VHlwZSA9ICdmbHVpZCc7XHJcbiAgY29udGVudFRyYW5zaXRpb25pbmcgPSBmYWxzZTtcclxuICBicmVhZGNydW1iczogVWlMYXlvdXRCcmVhZGNydW1iW10gPSBbXTtcclxuICB1c2VyRHJvcGRvd25PcGVuID0gZmFsc2U7XHJcbiAgaG5hdkRyb3Bkb3duSWQ6IHN0cmluZyB8IG51bGwgPSBudWxsO1xyXG5cclxuICBwcml2YXRlIG5hdkJhZGdlcyA9IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmcgfCBudW1iZXIgfCBudWxsPigpO1xyXG4gIHByaXZhdGUgbmF2VmlzaWJpbGl0eSA9IG5ldyBNYXA8c3RyaW5nLCBib29sZWFuPigpO1xyXG4gIHByaXZhdGUgYmFyVmlzaWJpbGl0eSA9IG5ldyBNYXA8c3RyaW5nLCBib29sZWFuPigpO1xyXG5cclxuICAvKiogUmVzb2x2ZWQgcGFnZSB0aXRsZSBmcm9tIHRoZSBsYXN0IGJyZWFkY3J1bWIuICovXHJcbiAgZ2V0IHBhZ2VUaXRsZSgpOiBzdHJpbmcge1xyXG4gICAgaWYgKCF0aGlzLmJyZWFkY3J1bWJzLmxlbmd0aCkgcmV0dXJuICcnO1xyXG4gICAgcmV0dXJuIHRoaXMuYnJlYWRjcnVtYnNbdGhpcy5icmVhZGNydW1icy5sZW5ndGggLSAxXS5sYWJlbDtcclxuICB9XHJcblxyXG4gIC8qKiBJdGVtcyBmbGFnZ2VkIGZvciB0aGUgbW9iaWxlIGJvdHRvbSBuYXZpZ2F0aW9uIGJhciAobWF4IDUpLiAqL1xyXG4gIGdldCBib3R0b21OYXZJdGVtcygpOiBVaU5hdkl0ZW1bXSB7XHJcbiAgICByZXR1cm4gdGhpcy5zY2hlbWEubmF2aWdhdGlvbi5pdGVtc1xyXG4gICAgICAuZmlsdGVyKChpKSA9PiBpLmJvdHRvbU5hdiAmJiAhdGhpcy5pc0l0ZW1IaWRkZW4oaSkgJiYgaS5yb3V0ZSlcclxuICAgICAgLnNvcnQoKGEsIGIpID0+IChhLmJvdHRvbU5hdk9yZGVyID8/IDk5KSAtIChiLmJvdHRvbU5hdk9yZGVyID8/IDk5KSlcclxuICAgICAgLnNsaWNlKDAsIDUpO1xyXG4gIH1cclxuXHJcbiAgZ2V0IG1vYmlsZU1vZGUoKTogc3RyaW5nIHtcclxuICAgIHJldHVybiB0aGlzLnNjaGVtYS5uYXZpZ2F0aW9uLm1vYmlsZU1vZGUgfHwgJ2JvdGgnO1xyXG4gIH1cclxuXHJcbiAgLyoqIFRvcGJhciBiYXJzIGZpbHRlcmVkIGJ5IHZpc2liaWxpdHkgKHNjaGVtYSBkZWZhdWx0ICsgc2VydmljZSBvdmVycmlkZXMpLiAqL1xyXG4gIGdldCB2aXNpYmxlQmFycygpOiBVaUJhckNvbmZpZ1tdIHtcclxuICAgIGlmICghdGhpcy5zY2hlbWEudG9wYmFyPy5iYXJzKSByZXR1cm4gW107XHJcbiAgICByZXR1cm4gdGhpcy5zY2hlbWEudG9wYmFyLmJhcnMuZmlsdGVyKChiYXIpID0+XHJcbiAgICAgIHRoaXMubGF5b3V0U2VydmljZS5pc0JhclZpc2libGUoYmFyLmlkLCBiYXIudmlzaWJsZSAhPT0gZmFsc2UpLFxyXG4gICAgKTtcclxuICB9XHJcblxyXG4gIG5nT25Jbml0KCk6IHZvaWQge1xyXG4gICAgdGhpcy5jdXJyZW50VXJsID0gdGhpcy5yb3V0ZXIudXJsO1xyXG4gICAgdGhpcy5hdXRvRXhwYW5kRnJvbVVybCh0aGlzLmN1cnJlbnRVcmwpO1xyXG5cclxuICAgIGlmICh0aGlzLnNjaGVtYS5tb2RlKSB7XHJcbiAgICAgIHRoaXMubGF5b3V0U2VydmljZS5fc2V0SW5pdGlhbExheW91dE1vZGUodGhpcy5zY2hlbWEubW9kZSk7XHJcbiAgICB9XHJcbiAgICBpZiAodGhpcy5zY2hlbWEuY29udGVudFR5cGUpIHtcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLl9zZXRJbml0aWFsQ29udGVudFR5cGUodGhpcy5zY2hlbWEuY29udGVudFR5cGUpO1xyXG4gICAgfVxyXG4gICAgaWYgKHRoaXMuc2NoZW1hLmZhYikge1xyXG4gICAgICB0aGlzLmxheW91dFNlcnZpY2UuX3NldERlZmF1bHRGYWJDb25maWcodGhpcy5zY2hlbWEuZmFiKTtcclxuICAgIH1cclxuICAgIGlmICh0aGlzLnNjaGVtYS5sb2FkZXI/LnNob3dPbkluaXQpIHtcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLnNob3dMb2FkZXIoKTtcclxuICAgIH1cclxuXHJcbiAgICB0aGlzLmJyZWFkY3J1bWJzID0gdGhpcy5jb21wdXRlQnJlYWRjcnVtYnModGhpcy5jdXJyZW50VXJsKTtcclxuXHJcbiAgICB0aGlzLnN1YnMucHVzaChcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLmxvYWRpbmckLnN1YnNjcmliZSgodikgPT4ge1xyXG4gICAgICAgIHRoaXMubG9hZGluZyA9IHY7XHJcbiAgICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICAgIH0pLFxyXG4gICAgICB0aGlzLmxheW91dFNlcnZpY2Uuc2lkZWJhck9wZW4kLnN1YnNjcmliZSgodikgPT4ge1xyXG4gICAgICAgIHRoaXMuc2lkZWJhck9wZW4gPSB2O1xyXG4gICAgICAgIHRoaXMuY2RyLm1hcmtGb3JDaGVjaygpO1xyXG4gICAgICB9KSxcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLnNpZGViYXJDb2xsYXBzZWQkLnN1YnNjcmliZSgodikgPT4ge1xyXG4gICAgICAgIHRoaXMuc2lkZWJhckNvbGxhcHNlZCA9IHY7XHJcbiAgICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICAgIH0pLFxyXG4gICAgICB0aGlzLmxheW91dFNlcnZpY2UuZmFiQ29uZmlnJC5zdWJzY3JpYmUoKGNvbmZpZykgPT4ge1xyXG4gICAgICAgIHRoaXMuY3VycmVudEZhYkNvbmZpZyA9IGNvbmZpZztcclxuICAgICAgICB0aGlzLmNkci5tYXJrRm9yQ2hlY2soKTtcclxuICAgICAgfSksXHJcbiAgICAgIHRoaXMubGF5b3V0U2VydmljZS5uYXZCYWRnZXMkLnN1YnNjcmliZSgobSkgPT4ge1xyXG4gICAgICAgIHRoaXMubmF2QmFkZ2VzID0gbTtcclxuICAgICAgICB0aGlzLmNkci5tYXJrRm9yQ2hlY2soKTtcclxuICAgICAgfSksXHJcbiAgICAgIHRoaXMubGF5b3V0U2VydmljZS5uYXZWaXNpYmlsaXR5JC5zdWJzY3JpYmUoKG0pID0+IHtcclxuICAgICAgICB0aGlzLm5hdlZpc2liaWxpdHkgPSBtO1xyXG4gICAgICAgIHRoaXMuY2RyLm1hcmtGb3JDaGVjaygpO1xyXG4gICAgICB9KSxcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLmxheW91dE1vZGUkLnN1YnNjcmliZSgobW9kZSkgPT4ge1xyXG4gICAgICAgIHRoaXMubGF5b3V0TW9kZSA9IG1vZGU7XHJcbiAgICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICAgIH0pLFxyXG4gICAgICB0aGlzLmxheW91dFNlcnZpY2UuY29udGVudFR5cGUkLnN1YnNjcmliZSgodHlwZSkgPT4ge1xyXG4gICAgICAgIHRoaXMuY29udGVudFR5cGUgPSB0eXBlO1xyXG4gICAgICAgIHRoaXMuY2RyLm1hcmtGb3JDaGVjaygpO1xyXG4gICAgICB9KSxcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLmJhclZpc2liaWxpdHkkLnN1YnNjcmliZSgobSkgPT4ge1xyXG4gICAgICAgIHRoaXMuYmFyVmlzaWJpbGl0eSA9IG07XHJcbiAgICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICAgIH0pLFxyXG4gICAgICB0aGlzLnJvdXRlci5ldmVudHNcclxuICAgICAgICAucGlwZShmaWx0ZXIoKGUpOiBlIGlzIE5hdmlnYXRpb25FbmQgPT4gZSBpbnN0YW5jZW9mIE5hdmlnYXRpb25FbmQpKVxyXG4gICAgICAgIC5zdWJzY3JpYmUoKGUpID0+IHtcclxuICAgICAgICAgIHRoaXMuY3VycmVudFVybCA9IGUudXJsQWZ0ZXJSZWRpcmVjdHM7XHJcbiAgICAgICAgICB0aGlzLmF1dG9FeHBhbmRGcm9tVXJsKHRoaXMuY3VycmVudFVybCk7XHJcbiAgICAgICAgICB0aGlzLmxheW91dFNlcnZpY2UuY2xvc2VNb2JpbGVTaWRlYmFyKCk7XHJcbiAgICAgICAgICB0aGlzLmJyZWFkY3J1bWJzID0gdGhpcy5jb21wdXRlQnJlYWRjcnVtYnModGhpcy5jdXJyZW50VXJsKTtcclxuICAgICAgICAgIHRoaXMuY29udGVudFRyYW5zaXRpb25pbmcgPSB0cnVlO1xyXG4gICAgICAgICAgdGhpcy51c2VyRHJvcGRvd25PcGVuID0gZmFsc2U7XHJcbiAgICAgICAgICB0aGlzLmhuYXZEcm9wZG93bklkID0gbnVsbDtcclxuXHJcbiAgICAgICAgICBpZiAodGhpcy5zY2hlbWEubG9hZGVyPy5zaG93T25Jbml0ICYmIHRoaXMubGF5b3V0U2VydmljZS5pc0xvYWRpbmcoKSkge1xyXG4gICAgICAgICAgICBjb25zdCBoaWRlID0gKCkgPT4gdGhpcy5sYXlvdXRTZXJ2aWNlLmhpZGVMb2FkZXIoKTtcclxuICAgICAgICAgICAgY29uc3QgbWluRHVyYXRpb24gPSB0aGlzLnNjaGVtYS5sb2FkZXIubWluRHVyYXRpb24gPz8gMDtcclxuICAgICAgICAgICAgaWYgKG1pbkR1cmF0aW9uID4gMCkge1xyXG4gICAgICAgICAgICAgIHNldFRpbWVvdXQoaGlkZSwgbWluRHVyYXRpb24pO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgIGhpZGUoKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgfVxyXG5cclxuICAgICAgICAgIHRoaXMuY2RyLm1hcmtGb3JDaGVjaygpO1xyXG4gICAgICAgIH0pLFxyXG4gICAgKTtcclxuICB9XHJcblxyXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xyXG4gICAgdGhpcy5zdWJzLmZvckVhY2goKHMpID0+IHMudW5zdWJzY3JpYmUoKSk7XHJcbiAgfVxyXG5cclxuICBASG9zdExpc3RlbmVyKCdkb2N1bWVudDprZXlkb3duLmVzY2FwZScpXHJcbiAgb25Fc2NhcGUoKTogdm9pZCB7XHJcbiAgICBpZiAodGhpcy5mYWJPcGVuKSB7XHJcbiAgICAgIHRoaXMuZmFiT3BlbiA9IGZhbHNlO1xyXG4gICAgfVxyXG4gICAgaWYgKHRoaXMudXNlckRyb3Bkb3duT3Blbikge1xyXG4gICAgICB0aGlzLnVzZXJEcm9wZG93bk9wZW4gPSBmYWxzZTtcclxuICAgIH1cclxuICAgIGlmICh0aGlzLmhuYXZEcm9wZG93bklkKSB7XHJcbiAgICAgIHRoaXMuaG5hdkRyb3Bkb3duSWQgPSBudWxsO1xyXG4gICAgfVxyXG4gICAgaWYgKHRoaXMuc2lkZWJhck9wZW4pIHtcclxuICAgICAgdGhpcy5sYXlvdXRTZXJ2aWNlLmNsb3NlTW9iaWxlU2lkZWJhcigpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgQEhvc3RMaXN0ZW5lcignZG9jdW1lbnQ6Y2xpY2snLCBbJyRldmVudCddKVxyXG4gIG9uRG9jdW1lbnRDbGljayhldmVudDogTW91c2VFdmVudCk6IHZvaWQge1xyXG4gICAgY29uc3QgdGFyZ2V0ID0gZXZlbnQudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xyXG4gICAgaWYgKHRoaXMudXNlckRyb3Bkb3duT3BlbiAmJiAhdGFyZ2V0LmNsb3Nlc3QoJy51aS1sYXlvdXRfX3VzZXItZHJvcGRvd24nKSkge1xyXG4gICAgICB0aGlzLnVzZXJEcm9wZG93bk9wZW4gPSBmYWxzZTtcclxuICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICB9XHJcbiAgICBpZiAodGhpcy5obmF2RHJvcGRvd25JZCAmJiAhdGFyZ2V0LmNsb3Nlc3QoJy51aS1sYXlvdXRfX2huYXYtZHJvcGRvd24nKSkge1xyXG4gICAgICB0aGlzLmhuYXZEcm9wZG93bklkID0gbnVsbDtcclxuICAgICAgdGhpcy5jZHIubWFya0ZvckNoZWNrKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvLyDilIDilIAgTmF2aWdhdGlvbiBoZWxwZXJzIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICBnZXRJdGVtc0ZvclNlY3Rpb24oc2VjdGlvbjogVWlOYXZTZWN0aW9uKTogVWlOYXZJdGVtW10ge1xyXG4gICAgY29uc3QgaWRTZXQgPSBuZXcgU2V0KHNlY3Rpb24uaXRlbUlkcyk7XHJcbiAgICByZXR1cm4gdGhpcy5zY2hlbWEubmF2aWdhdGlvbi5pdGVtcy5maWx0ZXIoKGkpID0+IGlkU2V0LmhhcyhpLmlkKSk7XHJcbiAgfVxyXG5cclxuICBpc0l0ZW1IaWRkZW4oaXRlbTogVWlOYXZJdGVtKTogYm9vbGVhbiB7XHJcbiAgICBjb25zdCBvdmVycmlkZSA9IHRoaXMubmF2VmlzaWJpbGl0eS5nZXQoaXRlbS5pZCk7XHJcbiAgICBpZiAob3ZlcnJpZGUgIT09IHVuZGVmaW5lZCkgcmV0dXJuICFvdmVycmlkZTtcclxuICAgIHJldHVybiBpdGVtLmhpZGRlbiA/PyBmYWxzZTtcclxuICB9XHJcblxyXG4gIGdldEl0ZW1CYWRnZShpdGVtOiBVaU5hdkl0ZW0pOiBzdHJpbmcgfCBudW1iZXIgfCBudWxsIHtcclxuICAgIHJldHVybiB0aGlzLm5hdkJhZGdlcy5nZXQoaXRlbS5pZCkgPz8gaXRlbS5iYWRnZSA/PyBudWxsO1xyXG4gIH1cclxuXHJcbiAgaXNQYXJlbnRBY3RpdmUoaXRlbTogVWlOYXZJdGVtKTogYm9vbGVhbiB7XHJcbiAgICBpZiAoIWl0ZW0ucm91dGUpIHJldHVybiBmYWxzZTtcclxuICAgIHJldHVybiB0aGlzLmN1cnJlbnRVcmwgPT09IGl0ZW0ucm91dGUgfHwgdGhpcy5jdXJyZW50VXJsLnN0YXJ0c1dpdGgoaXRlbS5yb3V0ZSArICcvJyk7XHJcbiAgfVxyXG5cclxuICBpc1JvdXRlQWN0aXZlKGl0ZW06IFVpTmF2SXRlbSk6IGJvb2xlYW4ge1xyXG4gICAgaWYgKCFpdGVtLnJvdXRlKSByZXR1cm4gZmFsc2U7XHJcbiAgICBpZiAoaXRlbS5yb3V0ZUFjdGl2ZUV4YWN0KSByZXR1cm4gdGhpcy5jdXJyZW50VXJsID09PSBpdGVtLnJvdXRlO1xyXG4gICAgcmV0dXJuIHRoaXMuY3VycmVudFVybCA9PT0gaXRlbS5yb3V0ZSB8fCB0aGlzLmN1cnJlbnRVcmwuc3RhcnRzV2l0aChpdGVtLnJvdXRlICsgJy8nKTtcclxuICB9XHJcblxyXG4gIHRvZ2dsZUV4cGFuZChpdGVtOiBVaU5hdkl0ZW0pOiB2b2lkIHtcclxuICAgIGlmICh0aGlzLmV4cGFuZGVkSWRzLmhhcyhpdGVtLmlkKSkge1xyXG4gICAgICB0aGlzLmV4cGFuZGVkSWRzLmRlbGV0ZShpdGVtLmlkKTtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIHRoaXMuZXhwYW5kZWRJZHMuYWRkKGl0ZW0uaWQpO1xyXG4gICAgICBpZiAoIXRoaXMuaXNQYXJlbnRBY3RpdmUoaXRlbSkgJiYgaXRlbS5jaGlsZHJlbj8ubGVuZ3RoKSB7XHJcbiAgICAgICAgY29uc3QgZmlyc3RDaGlsZCA9IGl0ZW0uY2hpbGRyZW5bMF0/Lml0ZW1zWzBdO1xyXG4gICAgICAgIGlmIChmaXJzdENoaWxkPy5yb3V0ZSkge1xyXG4gICAgICAgICAgdGhpcy5yb3V0ZXIubmF2aWdhdGVCeVVybChmaXJzdENoaWxkLnJvdXRlKTtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuICAgIH1cclxuICB9XHJcblxyXG4gIG9uTmF2Q2xpY2soKTogdm9pZCB7XHJcbiAgICB0aGlzLmxheW91dFNlcnZpY2UuY2xvc2VNb2JpbGVTaWRlYmFyKCk7XHJcbiAgfVxyXG5cclxuICAvLyDilIDilIAgRkFCIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICBvbkZhYk1haW5DbGljaygpOiB2b2lkIHtcclxuICAgIGlmICh0aGlzLmN1cnJlbnRGYWJDb25maWc/LnNlY29uZGFyeUFjdGlvbnM/Lmxlbmd0aCkge1xyXG4gICAgICB0aGlzLmZhYk9wZW4gPSAhdGhpcy5mYWJPcGVuO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgdGhpcy5jdXJyZW50RmFiQ29uZmlnPy5tYWluQWN0aW9uLmFjdGlvbj8uKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBvbkZhYkFjdGlvbihhY3Rpb246IFVpRmFiQWN0aW9uKTogdm9pZCB7XHJcbiAgICB0aGlzLmZhYk9wZW4gPSBmYWxzZTtcclxuICAgIGFjdGlvbi5hY3Rpb24/LigpO1xyXG4gIH1cclxuXHJcbiAgLy8g4pSA4pSAIFRvcGJhciBoZWxwZXJzIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICBnZXRUb3BiYXJOYXZJdGVtcyhiYXI6IFVpQmFyQ29uZmlnKTogVWlOYXZJdGVtW10ge1xyXG4gICAgcmV0dXJuIGJhci5uYXZpZ2F0aW9uPy5pdGVtcyA/PyB0aGlzLnNjaGVtYS5uYXZpZ2F0aW9uLml0ZW1zO1xyXG4gIH1cclxuXHJcbiAgdG9nZ2xlSG5hdkRyb3Bkb3duKGl0ZW1JZDogc3RyaW5nKTogdm9pZCB7XHJcbiAgICB0aGlzLmhuYXZEcm9wZG93bklkID0gdGhpcy5obmF2RHJvcGRvd25JZCA9PT0gaXRlbUlkID8gbnVsbCA6IGl0ZW1JZDtcclxuICB9XHJcblxyXG4gIGRpc21pc3NCYXIoYmFySWQ6IHN0cmluZyk6IHZvaWQge1xyXG4gICAgdGhpcy5sYXlvdXRTZXJ2aWNlLmhpZGVCYXIoYmFySWQpO1xyXG4gIH1cclxuXHJcbiAgb25Ecm9wZG93bkFjdGlvbihpdGVtOiB7IGFjdGlvbj86ICgpID0+IHZvaWQgfSk6IHZvaWQge1xyXG4gICAgdGhpcy51c2VyRHJvcGRvd25PcGVuID0gZmFsc2U7XHJcbiAgICBpdGVtLmFjdGlvbj8uKCk7XHJcbiAgfVxyXG5cclxuICAvLyDilIDilIAgQnJlYWRjcnVtYiBkZXJpdmF0aW9uIGZyb20gbmF2IHNjaGVtYSDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgLyoqXHJcbiAgICogRGVyaXZlcyBicmVhZGNydW1icyBieSBtYXRjaGluZyB0aGUgY3VycmVudCBVUkwgYWdhaW5zdCB0aGVcclxuICAgKiBuYXZpZ2F0aW9uIHNjaGVtYS4gV2Fsa3MgdG9wLWxldmVsIGl0ZW1zLCBzZWN0aW9ucywgYW5kIGNoaWxkcmVuXHJcbiAgICogdG8gYnVpbGQgYSB0cmFpbCBmcm9tIHJvb3QgdG8gY3VycmVudCBwYWdlLlxyXG4gICAqL1xyXG4gIHByaXZhdGUgY29tcHV0ZUJyZWFkY3J1bWJzKHVybDogc3RyaW5nKTogVWlMYXlvdXRCcmVhZGNydW1iW10ge1xyXG4gICAgY29uc3QgaXRlbXMgPSB0aGlzLnNjaGVtYS5uYXZpZ2F0aW9uLml0ZW1zO1xyXG4gICAgY29uc3Qgc2VjdGlvbnMgPSB0aGlzLnNjaGVtYS5uYXZpZ2F0aW9uLnNlY3Rpb25zO1xyXG5cclxuICAgIGZvciAoY29uc3QgaXRlbSBvZiBpdGVtcykge1xyXG4gICAgICBpZiAoaXRlbS50eXBlID09PSAnZGl2aWRlcicgfHwgaXRlbS50eXBlID09PSAnZXh0ZXJuYWwnKSBjb250aW51ZTtcclxuXHJcbiAgICAgIGlmIChpdGVtLmNoaWxkcmVuPy5sZW5ndGgpIHtcclxuICAgICAgICBmb3IgKGNvbnN0IGdyb3VwIG9mIGl0ZW0uY2hpbGRyZW4pIHtcclxuICAgICAgICAgIGZvciAoY29uc3QgY2hpbGQgb2YgZ3JvdXAuaXRlbXMpIHtcclxuICAgICAgICAgICAgaWYgKGNoaWxkLnJvdXRlICYmIHRoaXMudXJsTWF0Y2hlcyh1cmwsIGNoaWxkLnJvdXRlLCBjaGlsZC5yb3V0ZUFjdGl2ZUV4YWN0KSkge1xyXG4gICAgICAgICAgICAgIGNvbnN0IHRyYWlsOiBVaUxheW91dEJyZWFkY3J1bWJbXSA9IFtdO1xyXG4gICAgICAgICAgICAgIGNvbnN0IHNlY3Rpb25MYWJlbCA9IHRoaXMuZmluZFNlY3Rpb25MYWJlbChpdGVtLmlkLCBzZWN0aW9ucyk7XHJcbiAgICAgICAgICAgICAgaWYgKHNlY3Rpb25MYWJlbCkge1xyXG4gICAgICAgICAgICAgICAgdHJhaWwucHVzaCh7IGxhYmVsOiBzZWN0aW9uTGFiZWwsIGlzTGFzdDogZmFsc2UgfSk7XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgIHRyYWlsLnB1c2goeyBsYWJlbDogaXRlbS5sYWJlbCwgdXJsOiBpdGVtLnJvdXRlLCBpc0xhc3Q6IGZhbHNlIH0pO1xyXG4gICAgICAgICAgICAgIHRyYWlsLnB1c2goeyBsYWJlbDogY2hpbGQubGFiZWwsIGlzTGFzdDogdHJ1ZSB9KTtcclxuICAgICAgICAgICAgICByZXR1cm4gdHJhaWw7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIGlmIChpdGVtLnJvdXRlICYmIHRoaXMudXJsTWF0Y2hlcyh1cmwsIGl0ZW0ucm91dGUsIGl0ZW0ucm91dGVBY3RpdmVFeGFjdCkpIHtcclxuICAgICAgICBjb25zdCB0cmFpbDogVWlMYXlvdXRCcmVhZGNydW1iW10gPSBbXTtcclxuICAgICAgICBjb25zdCBzZWN0aW9uTGFiZWwgPSB0aGlzLmZpbmRTZWN0aW9uTGFiZWwoaXRlbS5pZCwgc2VjdGlvbnMpO1xyXG4gICAgICAgIGlmIChzZWN0aW9uTGFiZWwpIHtcclxuICAgICAgICAgIHRyYWlsLnB1c2goeyBsYWJlbDogc2VjdGlvbkxhYmVsLCBpc0xhc3Q6IGZhbHNlIH0pO1xyXG4gICAgICAgIH1cclxuICAgICAgICB0cmFpbC5wdXNoKHsgbGFiZWw6IGl0ZW0ubGFiZWwsIGlzTGFzdDogdHJ1ZSB9KTtcclxuICAgICAgICByZXR1cm4gdHJhaWw7XHJcbiAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gW107XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIHVybE1hdGNoZXMoY3VycmVudDogc3RyaW5nLCByb3V0ZTogc3RyaW5nLCBleGFjdD86IGJvb2xlYW4pOiBib29sZWFuIHtcclxuICAgIGlmIChleGFjdCkgcmV0dXJuIGN1cnJlbnQgPT09IHJvdXRlO1xyXG4gICAgcmV0dXJuIGN1cnJlbnQgPT09IHJvdXRlIHx8IGN1cnJlbnQuc3RhcnRzV2l0aChyb3V0ZSArICcvJyk7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGZpbmRTZWN0aW9uTGFiZWwoaXRlbUlkOiBzdHJpbmcsIHNlY3Rpb25zPzogVWlOYXZTZWN0aW9uW10pOiBzdHJpbmcgfCBudWxsIHtcclxuICAgIGlmICghc2VjdGlvbnM/Lmxlbmd0aCkgcmV0dXJuIG51bGw7XHJcbiAgICBjb25zdCBzZWN0aW9uID0gc2VjdGlvbnMuZmluZCgocykgPT4gcy5pdGVtSWRzLmluY2x1ZGVzKGl0ZW1JZCkpO1xyXG4gICAgcmV0dXJuIHNlY3Rpb24/LmxhYmVsID8/IG51bGw7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGF1dG9FeHBhbmRGcm9tVXJsKHVybDogc3RyaW5nKTogdm9pZCB7XHJcbiAgICBmb3IgKGNvbnN0IGl0ZW0gb2YgdGhpcy5zY2hlbWEubmF2aWdhdGlvbi5pdGVtcykge1xyXG4gICAgICBpZiAoaXRlbS5jaGlsZHJlbiAmJiBpdGVtLnJvdXRlICYmICh1cmwgPT09IGl0ZW0ucm91dGUgfHwgdXJsLnN0YXJ0c1dpdGgoaXRlbS5yb3V0ZSArICcvJykpKSB7XHJcbiAgICAgICAgdGhpcy5leHBhbmRlZElkcy5hZGQoaXRlbS5pZCk7XHJcbiAgICAgIH1cclxuICAgIH1cclxuICB9XHJcbn1cclxuIl19