@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,242 @@
1
+ import { inject, Injectable } from '@angular/core';
2
+ import { NavigationEnd, Router } from '@angular/router';
3
+ import { filter } from 'rxjs/operators';
4
+ import { UI_BREADCRUMB_LABEL_RESOLVER, } from './page-header.types';
5
+ import * as i0 from "@angular/core";
6
+ /**
7
+ * Service that builds a nested breadcrumb trail from Angular route configuration.
8
+ *
9
+ * Routes must define `data.id`, `data.title`, and optionally `data.breadcrumbs`
10
+ * (an array of parent route IDs) for the chain to resolve correctly.
11
+ *
12
+ * Supports lazy-loaded routes: the internal route map is rebuilt automatically
13
+ * whenever new routes are loaded via `NavigationEnd`.
14
+ *
15
+ * Label resolution for route parameters (`:param`) is delegated to an injectable
16
+ * `UiLabelResolverFn` via the `UI_BREADCRUMB_LABEL_RESOLVER` token.
17
+ *
18
+ * @usageNotes
19
+ * ### Route configuration
20
+ * ```typescript
21
+ * const routes: Routes = [
22
+ * {
23
+ * path: 'users',
24
+ * data: { id: 'users', title: 'Utenti' },
25
+ * children: [
26
+ * {
27
+ * path: ':userId',
28
+ * data: { id: 'user-detail', title: ':userId', breadcrumbs: ['users'] },
29
+ * component: UserDetailComponent,
30
+ * },
31
+ * ],
32
+ * },
33
+ * ];
34
+ * ```
35
+ *
36
+ * ### Custom label resolver
37
+ * ```typescript
38
+ * providers: [
39
+ * {
40
+ * provide: UI_BREADCRUMB_LABEL_RESOLVER,
41
+ * useValue: (name: string, value: string) =>
42
+ * name === 'userId' ? `User #${value}` : value,
43
+ * },
44
+ * ]
45
+ * ```
46
+ */
47
+ export class UiBreadcrumbService {
48
+ constructor() {
49
+ this.router = inject(Router);
50
+ this.labelResolver = inject(UI_BREADCRUMB_LABEL_RESOLVER);
51
+ /** @internal Map of route ID → breadcrumb configuration. */
52
+ this.breadcrumbMap = new Map();
53
+ /** @internal Tracks route count to detect lazy-loaded route additions. */
54
+ this.lastRouteConfigLength = 0;
55
+ /** @internal Last computed breadcrumbs for navigateBack. */
56
+ this.lastBreadcrumbs = [];
57
+ this.rebuildRouteMap();
58
+ this.router.events
59
+ .pipe(filter((event) => event instanceof NavigationEnd))
60
+ .subscribe(() => this.rebuildRouteMapIfNeeded());
61
+ }
62
+ /**
63
+ * Computes the breadcrumb trail for the currently active route.
64
+ *
65
+ * @param route - The root `ActivatedRoute` (typically injected in the component).
66
+ * @returns Ordered array of breadcrumb items from root to current route.
67
+ */
68
+ getBreadcrumbsForRoute(route) {
69
+ let deepestRoute = route;
70
+ while (deepestRoute.firstChild) {
71
+ deepestRoute = deepestRoute.firstChild;
72
+ }
73
+ let routeWithId = deepestRoute;
74
+ while (routeWithId && !routeWithId.snapshot.data['id']) {
75
+ routeWithId = routeWithId.parent;
76
+ }
77
+ const routeId = routeWithId?.snapshot.data['id'];
78
+ if (!routeId) {
79
+ this.lastBreadcrumbs = [];
80
+ return [];
81
+ }
82
+ const params = this.collectParamsFromPath(route);
83
+ const breadcrumbs = this.buildBreadcrumbChain(routeId, params);
84
+ this.lastBreadcrumbs = breadcrumbs;
85
+ return breadcrumbs;
86
+ }
87
+ /**
88
+ * Navigates to the previous breadcrumb (parent route).
89
+ * Falls back to `fallbackUrl` (default: `/`) when no parent exists.
90
+ */
91
+ navigateBack(fallbackUrl = '/') {
92
+ const prev = this.lastBreadcrumbs.length > 1 ? this.lastBreadcrumbs[this.lastBreadcrumbs.length - 2] : null;
93
+ const target = prev?.url ?? fallbackUrl;
94
+ this.router.navigateByUrl(target);
95
+ }
96
+ /** @internal Collects all route params from the full path (including parents). */
97
+ collectParamsFromPath(route) {
98
+ const params = {};
99
+ for (const r of route.pathFromRoot) {
100
+ Object.assign(params, r.snapshot.params);
101
+ }
102
+ return params;
103
+ }
104
+ // ─── Private helpers ─────────────────────────────────────────────
105
+ /** @internal Rebuilds the map only when new lazy routes appear. */
106
+ rebuildRouteMapIfNeeded() {
107
+ const currentLength = this.countRoutes(this.router.config);
108
+ if (currentLength !== this.lastRouteConfigLength) {
109
+ this.rebuildRouteMap();
110
+ }
111
+ }
112
+ /** @internal Recursively counts routes including lazy-loaded children. */
113
+ countRoutes(routes) {
114
+ let count = routes.length;
115
+ for (const route of routes) {
116
+ if (route.children) {
117
+ count += this.countRoutes(route.children);
118
+ }
119
+ if (route._loadedRoutes) {
120
+ count += this.countRoutes(route._loadedRoutes);
121
+ }
122
+ }
123
+ return count;
124
+ }
125
+ /** @internal Clears and rebuilds the full route → breadcrumb map. */
126
+ rebuildRouteMap() {
127
+ this.breadcrumbMap.clear();
128
+ this.buildRouteMap(this.router.config);
129
+ this.lastRouteConfigLength = this.countRoutes(this.router.config);
130
+ }
131
+ /** @internal Recursively walks route tree to build the breadcrumb map. */
132
+ buildRouteMap(routes, parentPath = '') {
133
+ for (const route of routes) {
134
+ const currentPath = parentPath + '/' + (route.path || '');
135
+ if (route.data?.['id']) {
136
+ const fullPath = this.normalizePath(currentPath);
137
+ this.breadcrumbMap.set(route.data['id'], {
138
+ id: route.data['id'],
139
+ path: fullPath,
140
+ label: route.data['title'] || route.path || route.data['id'],
141
+ parentIds: route.data['breadcrumbs'] || [],
142
+ metadata: route.data,
143
+ });
144
+ }
145
+ if (route.children) {
146
+ this.buildRouteMap(route.children, currentPath);
147
+ }
148
+ if (route._loadedRoutes) {
149
+ this.buildRouteMap(route._loadedRoutes, currentPath);
150
+ }
151
+ }
152
+ }
153
+ /** @internal Builds the full breadcrumb chain for a route by its ID. */
154
+ buildBreadcrumbChain(routeId, params) {
155
+ const config = this.breadcrumbMap.get(routeId);
156
+ if (!config)
157
+ return [];
158
+ this.saveFullPath(routeId, this.router.url);
159
+ const parentChain = this.getParentChain(config.parentIds, params);
160
+ const breadcrumbs = [...parentChain];
161
+ breadcrumbs.push({
162
+ label: this.resolveLabel(config.label, params),
163
+ url: this.stripQueryParams(this.router.url),
164
+ isLast: true,
165
+ });
166
+ return breadcrumbs;
167
+ }
168
+ /** @internal Resolves parent breadcrumbs from ordered parent IDs. */
169
+ getParentChain(parentIds, params) {
170
+ if (!parentIds?.length)
171
+ return [];
172
+ return parentIds
173
+ .map((id) => {
174
+ const config = this.breadcrumbMap.get(id);
175
+ if (!config)
176
+ return null;
177
+ return {
178
+ label: this.resolveLabel(config.label, params),
179
+ url: this.resolveUrl(config.path, params),
180
+ isLast: false,
181
+ };
182
+ })
183
+ .filter(Boolean);
184
+ }
185
+ /**
186
+ * @internal Resolves `:param` placeholders in a label using
187
+ * the injected label resolver function.
188
+ */
189
+ resolveLabel(label, params) {
190
+ if (!label.includes(':'))
191
+ return label;
192
+ let resolved = label;
193
+ for (const [key, value] of Object.entries(params)) {
194
+ const readableValue = this.labelResolver(key, value);
195
+ resolved = resolved.replace(`:${key}`, readableValue);
196
+ }
197
+ return resolved;
198
+ }
199
+ /** @internal Resolves `:param` placeholders in a path and creates a URL. */
200
+ resolveUrl(path, params) {
201
+ let processedPath = path;
202
+ if (path.includes(':')) {
203
+ const segments = path.split('/');
204
+ const processedSegments = segments.map((segment) => {
205
+ if (segment.startsWith(':')) {
206
+ const paramName = segment.substring(1);
207
+ return params[paramName] || segment;
208
+ }
209
+ return segment;
210
+ });
211
+ processedPath = processedSegments.join('/');
212
+ }
213
+ try {
214
+ return this.router.createUrlTree([processedPath]).toString();
215
+ }
216
+ catch {
217
+ return processedPath;
218
+ }
219
+ }
220
+ /** @internal Normalizes path by collapsing double slashes and trimming trailing slash. */
221
+ normalizePath(path) {
222
+ return path.replace(/\/+/g, '/').replace(/\/$/, '');
223
+ }
224
+ /** @internal Removes query parameters from a URL. */
225
+ stripQueryParams(url) {
226
+ return url.split('?')[0];
227
+ }
228
+ /** @internal Caches the fully resolved URL for a route ID. */
229
+ saveFullPath(routeId, fullPath) {
230
+ const config = this.breadcrumbMap.get(routeId);
231
+ if (config) {
232
+ config.fullPath = this.stripQueryParams(fullPath);
233
+ }
234
+ }
235
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBreadcrumbService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
236
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBreadcrumbService, providedIn: 'root' }); }
237
+ }
238
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBreadcrumbService, decorators: [{
239
+ type: Injectable,
240
+ args: [{ providedIn: 'root' }]
241
+ }], ctorParameters: () => [] });
242
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"breadcrumb.service.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/page-header/breadcrumb.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\r\nimport { ActivatedRoute, NavigationEnd, Router, Routes } from '@angular/router';\r\nimport { filter } from 'rxjs/operators';\r\nimport {\r\n  UiBreadcrumbConfig,\r\n  UiBreadcrumbItem,\r\n  UiLabelResolverFn,\r\n  UI_BREADCRUMB_LABEL_RESOLVER,\r\n} from './page-header.types';\r\n\r\n/**\r\n * Service that builds a nested breadcrumb trail from Angular route configuration.\r\n *\r\n * Routes must define `data.id`, `data.title`, and optionally `data.breadcrumbs`\r\n * (an array of parent route IDs) for the chain to resolve correctly.\r\n *\r\n * Supports lazy-loaded routes: the internal route map is rebuilt automatically\r\n * whenever new routes are loaded via `NavigationEnd`.\r\n *\r\n * Label resolution for route parameters (`:param`) is delegated to an injectable\r\n * `UiLabelResolverFn` via the `UI_BREADCRUMB_LABEL_RESOLVER` token.\r\n *\r\n * @usageNotes\r\n * ### Route configuration\r\n * ```typescript\r\n * const routes: Routes = [\r\n *   {\r\n *     path: 'users',\r\n *     data: { id: 'users', title: 'Utenti' },\r\n *     children: [\r\n *       {\r\n *         path: ':userId',\r\n *         data: { id: 'user-detail', title: ':userId', breadcrumbs: ['users'] },\r\n *         component: UserDetailComponent,\r\n *       },\r\n *     ],\r\n *   },\r\n * ];\r\n * ```\r\n *\r\n * ### Custom label resolver\r\n * ```typescript\r\n * providers: [\r\n *   {\r\n *     provide: UI_BREADCRUMB_LABEL_RESOLVER,\r\n *     useValue: (name: string, value: string) =>\r\n *       name === 'userId' ? `User #${value}` : value,\r\n *   },\r\n * ]\r\n * ```\r\n */\r\n@Injectable({ providedIn: 'root' })\r\nexport class UiBreadcrumbService {\r\n  private readonly router = inject(Router);\r\n  private readonly labelResolver = inject(UI_BREADCRUMB_LABEL_RESOLVER);\r\n\r\n  /** @internal Map of route ID → breadcrumb configuration. */\r\n  private breadcrumbMap = new Map<string, UiBreadcrumbConfig>();\r\n  /** @internal Tracks route count to detect lazy-loaded route additions. */\r\n  private lastRouteConfigLength = 0;\r\n  /** @internal Last computed breadcrumbs for navigateBack. */\r\n  private lastBreadcrumbs: UiBreadcrumbItem[] = [];\r\n\r\n  constructor() {\r\n    this.rebuildRouteMap();\r\n\r\n    this.router.events\r\n      .pipe(filter((event) => event instanceof NavigationEnd))\r\n      .subscribe(() => this.rebuildRouteMapIfNeeded());\r\n  }\r\n\r\n  /**\r\n   * Computes the breadcrumb trail for the currently active route.\r\n   *\r\n   * @param route - The root `ActivatedRoute` (typically injected in the component).\r\n   * @returns Ordered array of breadcrumb items from root to current route.\r\n   */\r\n  getBreadcrumbsForRoute(route: ActivatedRoute): UiBreadcrumbItem[] {\r\n    let deepestRoute = route;\r\n    while (deepestRoute.firstChild) {\r\n      deepestRoute = deepestRoute.firstChild;\r\n    }\r\n\r\n    let routeWithId: ActivatedRoute | null = deepestRoute;\r\n    while (routeWithId && !routeWithId.snapshot.data['id']) {\r\n      routeWithId = routeWithId.parent;\r\n    }\r\n    const routeId = routeWithId?.snapshot.data['id'];\r\n    if (!routeId) {\r\n      this.lastBreadcrumbs = [];\r\n      return [];\r\n    }\r\n\r\n    const params = this.collectParamsFromPath(route);\r\n    const breadcrumbs = this.buildBreadcrumbChain(routeId, params);\r\n    this.lastBreadcrumbs = breadcrumbs;\r\n    return breadcrumbs;\r\n  }\r\n\r\n  /**\r\n   * Navigates to the previous breadcrumb (parent route).\r\n   * Falls back to `fallbackUrl` (default: `/`) when no parent exists.\r\n   */\r\n  navigateBack(fallbackUrl = '/'): void {\r\n    const prev = this.lastBreadcrumbs.length > 1 ? this.lastBreadcrumbs[this.lastBreadcrumbs.length - 2] : null;\r\n    const target = prev?.url ?? fallbackUrl;\r\n    this.router.navigateByUrl(target);\r\n  }\r\n\r\n  /** @internal Collects all route params from the full path (including parents). */\r\n  private collectParamsFromPath(route: ActivatedRoute): Record<string, string> {\r\n    const params: Record<string, string> = {};\r\n    for (const r of route.pathFromRoot) {\r\n      Object.assign(params, r.snapshot.params);\r\n    }\r\n    return params;\r\n  }\r\n\r\n  // ─── Private helpers ─────────────────────────────────────────────\r\n\r\n  /** @internal Rebuilds the map only when new lazy routes appear. */\r\n  private rebuildRouteMapIfNeeded(): void {\r\n    const currentLength = this.countRoutes(this.router.config);\r\n    if (currentLength !== this.lastRouteConfigLength) {\r\n      this.rebuildRouteMap();\r\n    }\r\n  }\r\n\r\n  /** @internal Recursively counts routes including lazy-loaded children. */\r\n  private countRoutes(routes: Routes): number {\r\n    let count = routes.length;\r\n    for (const route of routes) {\r\n      if (route.children) {\r\n        count += this.countRoutes(route.children);\r\n      }\r\n      if ((route as any)._loadedRoutes) {\r\n        count += this.countRoutes((route as any)._loadedRoutes);\r\n      }\r\n    }\r\n    return count;\r\n  }\r\n\r\n  /** @internal Clears and rebuilds the full route → breadcrumb map. */\r\n  private rebuildRouteMap(): void {\r\n    this.breadcrumbMap.clear();\r\n    this.buildRouteMap(this.router.config);\r\n    this.lastRouteConfigLength = this.countRoutes(this.router.config);\r\n  }\r\n\r\n  /** @internal Recursively walks route tree to build the breadcrumb map. */\r\n  private buildRouteMap(routes: Routes, parentPath = ''): void {\r\n    for (const route of routes) {\r\n      const currentPath = parentPath + '/' + (route.path || '');\r\n\r\n      if (route.data?.['id']) {\r\n        const fullPath = this.normalizePath(currentPath);\r\n        this.breadcrumbMap.set(route.data['id'], {\r\n          id: route.data['id'],\r\n          path: fullPath,\r\n          label: route.data['title'] || route.path || route.data['id'],\r\n          parentIds: route.data['breadcrumbs'] || [],\r\n          metadata: route.data,\r\n        });\r\n      }\r\n\r\n      if (route.children) {\r\n        this.buildRouteMap(route.children, currentPath);\r\n      }\r\n      if ((route as any)._loadedRoutes) {\r\n        this.buildRouteMap((route as any)._loadedRoutes, currentPath);\r\n      }\r\n    }\r\n  }\r\n\r\n  /** @internal Builds the full breadcrumb chain for a route by its ID. */\r\n  private buildBreadcrumbChain(routeId: string, params: Record<string, string>): UiBreadcrumbItem[] {\r\n    const config = this.breadcrumbMap.get(routeId);\r\n    if (!config) return [];\r\n\r\n    this.saveFullPath(routeId, this.router.url);\r\n\r\n    const parentChain = this.getParentChain(config.parentIds, params);\r\n    const breadcrumbs: UiBreadcrumbItem[] = [...parentChain];\r\n\r\n    breadcrumbs.push({\r\n      label: this.resolveLabel(config.label, params),\r\n      url: this.stripQueryParams(this.router.url),\r\n      isLast: true,\r\n    });\r\n\r\n    return breadcrumbs;\r\n  }\r\n\r\n  /** @internal Resolves parent breadcrumbs from ordered parent IDs. */\r\n  private getParentChain(parentIds: string[], params: Record<string, string>): UiBreadcrumbItem[] {\r\n    if (!parentIds?.length) return [];\r\n\r\n    return parentIds\r\n      .map((id) => {\r\n        const config = this.breadcrumbMap.get(id);\r\n        if (!config) return null;\r\n\r\n        return {\r\n          label: this.resolveLabel(config.label, params),\r\n          url: this.resolveUrl(config.path, params),\r\n          isLast: false,\r\n        };\r\n      })\r\n      .filter(Boolean) as UiBreadcrumbItem[];\r\n  }\r\n\r\n  /**\r\n   * @internal Resolves `:param` placeholders in a label using\r\n   * the injected label resolver function.\r\n   */\r\n  private resolveLabel(label: string, params: Record<string, string>): string {\r\n    if (!label.includes(':')) return label;\r\n\r\n    let resolved = label;\r\n    for (const [key, value] of Object.entries(params)) {\r\n      const readableValue = this.labelResolver(key, value);\r\n      resolved = resolved.replace(`:${key}`, readableValue);\r\n    }\r\n    return resolved;\r\n  }\r\n\r\n  /** @internal Resolves `:param` placeholders in a path and creates a URL. */\r\n  private resolveUrl(path: string, params: Record<string, string>): string {\r\n    let processedPath = path;\r\n\r\n    if (path.includes(':')) {\r\n      const segments = path.split('/');\r\n      const processedSegments = segments.map((segment) => {\r\n        if (segment.startsWith(':')) {\r\n          const paramName = segment.substring(1);\r\n          return params[paramName] || segment;\r\n        }\r\n        return segment;\r\n      });\r\n      processedPath = processedSegments.join('/');\r\n    }\r\n\r\n    try {\r\n      return this.router.createUrlTree([processedPath]).toString();\r\n    } catch {\r\n      return processedPath;\r\n    }\r\n  }\r\n\r\n  /** @internal Normalizes path by collapsing double slashes and trimming trailing slash. */\r\n  private normalizePath(path: string): string {\r\n    return path.replace(/\\/+/g, '/').replace(/\\/$/, '');\r\n  }\r\n\r\n  /** @internal Removes query parameters from a URL. */\r\n  private stripQueryParams(url: string): string {\r\n    return url.split('?')[0];\r\n  }\r\n\r\n  /** @internal Caches the fully resolved URL for a route ID. */\r\n  private saveFullPath(routeId: string, fullPath: string): void {\r\n    const config = this.breadcrumbMap.get(routeId);\r\n    if (config) {\r\n      config.fullPath = this.stripQueryParams(fullPath);\r\n    }\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,20 @@
1
+ /**
2
+ * ng-ui-system — Page Header entry point.
3
+ *
4
+ * @example
5
+ * ```typescript
6
+ * import {
7
+ * UiPageHeaderComponent,
8
+ * UiBreadcrumbService,
9
+ * UiBreadcrumbItem,
10
+ * UI_BREADCRUMB_LABEL_RESOLVER,
11
+ * } from 'ng-ui-system';
12
+ * ```
13
+ */
14
+ // Components
15
+ export { UiPageHeaderComponent } from './page-header.component';
16
+ // Services
17
+ export { UiBreadcrumbService } from './breadcrumb.service';
18
+ // Types & Tokens
19
+ export { UI_BREADCRUMB_LABEL_RESOLVER, } from './page-header.types';
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL3BhZ2UtaGVhZGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7R0FZRztBQUVILGFBQWE7QUFDYixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUVoRSxXQUFXO0FBQ1gsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFFM0QsaUJBQWlCO0FBQ2pCLE9BQU8sRUFJTCw0QkFBNEIsR0FDN0IsTUFBTSxxQkFBcUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBuZy11aS1zeXN0ZW0g4oCUIFBhZ2UgSGVhZGVyIGVudHJ5IHBvaW50LlxyXG4gKlxyXG4gKiBAZXhhbXBsZVxyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGltcG9ydCB7XHJcbiAqICAgVWlQYWdlSGVhZGVyQ29tcG9uZW50LFxyXG4gKiAgIFVpQnJlYWRjcnVtYlNlcnZpY2UsXHJcbiAqICAgVWlCcmVhZGNydW1iSXRlbSxcclxuICogICBVSV9CUkVBRENSVU1CX0xBQkVMX1JFU09MVkVSLFxyXG4gKiB9IGZyb20gJ25nLXVpLXN5c3RlbSc7XHJcbiAqIGBgYFxyXG4gKi9cclxuXHJcbi8vIENvbXBvbmVudHNcclxuZXhwb3J0IHsgVWlQYWdlSGVhZGVyQ29tcG9uZW50IH0gZnJvbSAnLi9wYWdlLWhlYWRlci5jb21wb25lbnQnO1xyXG5cclxuLy8gU2VydmljZXNcclxuZXhwb3J0IHsgVWlCcmVhZGNydW1iU2VydmljZSB9IGZyb20gJy4vYnJlYWRjcnVtYi5zZXJ2aWNlJztcclxuXHJcbi8vIFR5cGVzICYgVG9rZW5zXHJcbmV4cG9ydCB7XHJcbiAgVWlCcmVhZGNydW1iSXRlbSxcclxuICBVaUJyZWFkY3J1bWJDb25maWcsXHJcbiAgVWlMYWJlbFJlc29sdmVyRm4sXHJcbiAgVUlfQlJFQURDUlVNQl9MQUJFTF9SRVNPTFZFUixcclxufSBmcm9tICcuL3BhZ2UtaGVhZGVyLnR5cGVzJztcclxuIl19
@@ -0,0 +1,243 @@
1
+ import { ChangeDetectorRef, Component, DestroyRef, inject, Input, ChangeDetectionStrategy, ViewEncapsulation, } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { ActivatedRoute, NavigationEnd, Router, RouterLink } from '@angular/router';
4
+ import { Title } from '@angular/platform-browser';
5
+ import { filter } from 'rxjs/operators';
6
+ import { LucideAngularModule } from 'lucide-angular';
7
+ import { UiBreadcrumbService } from './breadcrumb.service';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "lucide-angular";
10
+ /**
11
+ * Page header component that renders a breadcrumb navigation trail
12
+ * and a page title. Integrates with `UiBreadcrumbService` to
13
+ * automatically derive breadcrumbs from Angular route configuration.
14
+ *
15
+ * The breadcrumb trail is generated from route `data` properties:
16
+ * - `data.id` — unique route identifier
17
+ * - `data.title` — display label
18
+ * - `data.breadcrumbs` — ordered array of parent route IDs
19
+ *
20
+ * @selector ui-page-header
21
+ *
22
+ * @example
23
+ * ```html
24
+ * <!-- Auto breadcrumbs from route data -->
25
+ * <ui-page-header />
26
+ *
27
+ * <!-- With explicit title override -->
28
+ * <ui-page-header title="Custom Title" />
29
+ *
30
+ * <!-- With document title update -->
31
+ * <ui-page-header [updateDocumentTitle]="true" titleSuffix=" | My App" />
32
+ * ```
33
+ */
34
+ export class UiPageHeaderComponent {
35
+ constructor() {
36
+ /** Override the auto-detected page title. When empty, title is derived from the last breadcrumb. */
37
+ this.title = '';
38
+ /** Route path for the Home breadcrumb link. */
39
+ this.homeRoute = '/';
40
+ /** Whether to display the Home link at the start of the trail. */
41
+ this.showHome = true;
42
+ /** When `true`, updates `document.title` on each navigation. */
43
+ this.updateDocumentTitle = false;
44
+ /** Suffix appended to `document.title` (e.g. `' | My App'`). Only used when `updateDocumentTitle` is `true`. */
45
+ this.titleSuffix = '';
46
+ /** @internal */
47
+ this.breadcrumbs = [];
48
+ this.router = inject(Router);
49
+ this.activatedRoute = inject(ActivatedRoute);
50
+ this.titleService = inject(Title);
51
+ this.breadcrumbService = inject(UiBreadcrumbService);
52
+ this.destroyRef = inject(DestroyRef);
53
+ this.cdr = inject(ChangeDetectorRef);
54
+ }
55
+ /** Resolved page title: explicit `title` input wins, otherwise last breadcrumb label. */
56
+ get displayTitle() {
57
+ if (this.title)
58
+ return this.title;
59
+ if (this.breadcrumbs.length > 0) {
60
+ return this.breadcrumbs[this.breadcrumbs.length - 1].label;
61
+ }
62
+ return '';
63
+ }
64
+ ngOnInit() {
65
+ this.refresh();
66
+ this.router.events
67
+ .pipe(filter((event) => event instanceof NavigationEnd), takeUntilDestroyed(this.destroyRef))
68
+ .subscribe(() => this.refresh());
69
+ }
70
+ /**
71
+ * Navigates to the previous breadcrumb (parent route).
72
+ * @param fallbackUrl - URL to navigate to when no parent exists (default: homeRoute).
73
+ */
74
+ navigateBack(fallbackUrl) {
75
+ this.breadcrumbService.navigateBack(fallbackUrl ?? this.homeRoute);
76
+ }
77
+ /** @internal */
78
+ refresh() {
79
+ this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
80
+ if (this.updateDocumentTitle) {
81
+ const title = this.displayTitle;
82
+ if (title) {
83
+ this.titleService.setTitle(`${title}${this.titleSuffix}`);
84
+ }
85
+ }
86
+ this.cdr.markForCheck();
87
+ }
88
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiPageHeaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
89
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiPageHeaderComponent, isStandalone: true, selector: "ui-page-header", inputs: { title: "title", homeRoute: "homeRoute", showHome: "showHome", updateDocumentTitle: "updateDocumentTitle", titleSuffix: "titleSuffix" }, host: { classAttribute: "ui-page-header-host" }, ngImport: i0, template: `
90
+ <header class="ui-page-header">
91
+ <nav aria-label="Breadcrumb" class="ui-page-header__nav">
92
+ <ol
93
+ class="ui-page-header__breadcrumb"
94
+ itemscope
95
+ itemtype="https://schema.org/BreadcrumbList"
96
+ >
97
+ @if (showHome) {
98
+ <li
99
+ class="ui-page-header__item"
100
+ itemprop="itemListElement"
101
+ itemscope
102
+ itemtype="https://schema.org/ListItem"
103
+ >
104
+ <a
105
+ itemprop="item"
106
+ [routerLink]="homeRoute"
107
+ class="ui-page-header__link"
108
+ aria-label="Home"
109
+ >
110
+ <lucide-icon name="home" [size]="14" aria-hidden="true" />
111
+ <span itemprop="name" class="ui-page-header__text">Home</span>
112
+ </a>
113
+ <meta itemprop="position" content="1" />
114
+ <span class="ui-page-header__separator" aria-hidden="true">
115
+ <lucide-icon name="chevron-right" [size]="14" />
116
+ </span>
117
+ </li>
118
+ }
119
+ @for (crumb of breadcrumbs; track crumb.url; let i = $index) {
120
+ <li
121
+ class="ui-page-header__item"
122
+ [class.ui-page-header__item--current]="crumb.isLast"
123
+ itemprop="itemListElement"
124
+ itemscope
125
+ itemtype="https://schema.org/ListItem"
126
+ >
127
+ @if (!crumb.isLast) {
128
+ <a
129
+ itemprop="item"
130
+ [routerLink]="crumb.url"
131
+ class="ui-page-header__link"
132
+ >
133
+ <span itemprop="name" class="ui-page-header__text">{{ crumb.label }}</span>
134
+ </a>
135
+ <span class="ui-page-header__separator" aria-hidden="true">
136
+ <lucide-icon name="chevron-right" [size]="14" />
137
+ </span>
138
+ } @else {
139
+ <span
140
+ itemprop="name"
141
+ class="ui-page-header__text ui-page-header__text--current"
142
+ aria-current="page"
143
+ >
144
+ {{ crumb.label }}
145
+ </span>
146
+ }
147
+ <meta itemprop="position" [attr.content]="showHome ? i + 2 : i + 1" />
148
+ </li>
149
+ }
150
+ </ol>
151
+ </nav>
152
+
153
+ <h1 class="ui-page-header__title" id="page-title">
154
+ {{ displayTitle }}
155
+ </h1>
156
+ </header>
157
+ `, isInline: true, styles: [".ui-page-header-host{display:block}.ui-page-header{margin-bottom:var(--ui-spacing-5)}.ui-page-header__nav{margin-bottom:var(--ui-spacing-2)}.ui-page-header__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-page-header__item{display:inline-flex;align-items:center;gap:var(--ui-spacing-1)}a.ui-page-header__link lucide-icon{transform:translateY(2px)!important}span.ui-page-header__separator lucide-icon{transform:translateY(3px)!important}.ui-page-header__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-page-header__link:hover{color:var(--ui-color-primary-hover);text-decoration:underline}.ui-page-header__link:focus{outline:none}.ui-page-header__link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-page-header__text{white-space:nowrap}.ui-page-header__text--current{font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text)}.ui-page-header__separator{display:inline-flex;align-items:center;color:var(--ui-color-text-muted);flex-shrink:0}.ui-page-header__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-page-header__title:focus{outline:none}@media (min-width: 640px){.ui-page-header__title{font-size:var(--ui-font-size-xl)}.ui-page-header__breadcrumb{font-size:var(--ui-font-size-xs)}}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { 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 }); }
158
+ }
159
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiPageHeaderComponent, decorators: [{
160
+ type: Component,
161
+ args: [{ selector: 'ui-page-header', standalone: true, imports: [RouterLink, LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
162
+ class: 'ui-page-header-host',
163
+ }, template: `
164
+ <header class="ui-page-header">
165
+ <nav aria-label="Breadcrumb" class="ui-page-header__nav">
166
+ <ol
167
+ class="ui-page-header__breadcrumb"
168
+ itemscope
169
+ itemtype="https://schema.org/BreadcrumbList"
170
+ >
171
+ @if (showHome) {
172
+ <li
173
+ class="ui-page-header__item"
174
+ itemprop="itemListElement"
175
+ itemscope
176
+ itemtype="https://schema.org/ListItem"
177
+ >
178
+ <a
179
+ itemprop="item"
180
+ [routerLink]="homeRoute"
181
+ class="ui-page-header__link"
182
+ aria-label="Home"
183
+ >
184
+ <lucide-icon name="home" [size]="14" aria-hidden="true" />
185
+ <span itemprop="name" class="ui-page-header__text">Home</span>
186
+ </a>
187
+ <meta itemprop="position" content="1" />
188
+ <span class="ui-page-header__separator" aria-hidden="true">
189
+ <lucide-icon name="chevron-right" [size]="14" />
190
+ </span>
191
+ </li>
192
+ }
193
+ @for (crumb of breadcrumbs; track crumb.url; let i = $index) {
194
+ <li
195
+ class="ui-page-header__item"
196
+ [class.ui-page-header__item--current]="crumb.isLast"
197
+ itemprop="itemListElement"
198
+ itemscope
199
+ itemtype="https://schema.org/ListItem"
200
+ >
201
+ @if (!crumb.isLast) {
202
+ <a
203
+ itemprop="item"
204
+ [routerLink]="crumb.url"
205
+ class="ui-page-header__link"
206
+ >
207
+ <span itemprop="name" class="ui-page-header__text">{{ crumb.label }}</span>
208
+ </a>
209
+ <span class="ui-page-header__separator" aria-hidden="true">
210
+ <lucide-icon name="chevron-right" [size]="14" />
211
+ </span>
212
+ } @else {
213
+ <span
214
+ itemprop="name"
215
+ class="ui-page-header__text ui-page-header__text--current"
216
+ aria-current="page"
217
+ >
218
+ {{ crumb.label }}
219
+ </span>
220
+ }
221
+ <meta itemprop="position" [attr.content]="showHome ? i + 2 : i + 1" />
222
+ </li>
223
+ }
224
+ </ol>
225
+ </nav>
226
+
227
+ <h1 class="ui-page-header__title" id="page-title">
228
+ {{ displayTitle }}
229
+ </h1>
230
+ </header>
231
+ `, styles: [".ui-page-header-host{display:block}.ui-page-header{margin-bottom:var(--ui-spacing-5)}.ui-page-header__nav{margin-bottom:var(--ui-spacing-2)}.ui-page-header__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-page-header__item{display:inline-flex;align-items:center;gap:var(--ui-spacing-1)}a.ui-page-header__link lucide-icon{transform:translateY(2px)!important}span.ui-page-header__separator lucide-icon{transform:translateY(3px)!important}.ui-page-header__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-page-header__link:hover{color:var(--ui-color-primary-hover);text-decoration:underline}.ui-page-header__link:focus{outline:none}.ui-page-header__link:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-page-header__text{white-space:nowrap}.ui-page-header__text--current{font-weight:var(--ui-font-weight-medium);color:var(--ui-color-text)}.ui-page-header__separator{display:inline-flex;align-items:center;color:var(--ui-color-text-muted);flex-shrink:0}.ui-page-header__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-page-header__title:focus{outline:none}@media (min-width: 640px){.ui-page-header__title{font-size:var(--ui-font-size-xl)}.ui-page-header__breadcrumb{font-size:var(--ui-font-size-xs)}}\n"] }]
232
+ }], propDecorators: { title: [{
233
+ type: Input
234
+ }], homeRoute: [{
235
+ type: Input
236
+ }], showHome: [{
237
+ type: Input
238
+ }], updateDocumentTitle: [{
239
+ type: Input
240
+ }], titleSuffix: [{
241
+ type: Input
242
+ }] } });
243
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-header.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/page-header/page-header.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectorRef,\r\n  Component,\r\n  DestroyRef,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  ChangeDetectionStrategy,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { ActivatedRoute, NavigationEnd, Router, RouterLink } from '@angular/router';\r\nimport { Title } from '@angular/platform-browser';\r\nimport { filter } from 'rxjs/operators';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiBreadcrumbService } from './breadcrumb.service';\r\nimport { UiBreadcrumbItem } from './page-header.types';\r\n\r\n/**\r\n * Page header component that renders a breadcrumb navigation trail\r\n * and a page title. Integrates with `UiBreadcrumbService` to\r\n * automatically derive breadcrumbs from Angular route configuration.\r\n *\r\n * The breadcrumb trail is generated from route `data` properties:\r\n * - `data.id` — unique route identifier\r\n * - `data.title` — display label\r\n * - `data.breadcrumbs` — ordered array of parent route IDs\r\n *\r\n * @selector ui-page-header\r\n *\r\n * @example\r\n * ```html\r\n * <!-- Auto breadcrumbs from route data -->\r\n * <ui-page-header />\r\n *\r\n * <!-- With explicit title override -->\r\n * <ui-page-header title=\"Custom Title\" />\r\n *\r\n * <!-- With document title update -->\r\n * <ui-page-header [updateDocumentTitle]=\"true\" titleSuffix=\" | My App\" />\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-page-header',\r\n  standalone: true,\r\n  imports: [RouterLink, LucideAngularModule],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-page-header-host',\r\n  },\r\n  template: `\r\n    <header class=\"ui-page-header\">\r\n      <nav aria-label=\"Breadcrumb\" class=\"ui-page-header__nav\">\r\n        <ol\r\n          class=\"ui-page-header__breadcrumb\"\r\n          itemscope\r\n          itemtype=\"https://schema.org/BreadcrumbList\"\r\n        >\r\n          @if (showHome) {\r\n            <li\r\n              class=\"ui-page-header__item\"\r\n              itemprop=\"itemListElement\"\r\n              itemscope\r\n              itemtype=\"https://schema.org/ListItem\"\r\n            >\r\n              <a\r\n                itemprop=\"item\"\r\n                [routerLink]=\"homeRoute\"\r\n                class=\"ui-page-header__link\"\r\n                aria-label=\"Home\"\r\n              >\r\n                <lucide-icon name=\"home\" [size]=\"14\" aria-hidden=\"true\" />\r\n                <span itemprop=\"name\" class=\"ui-page-header__text\">Home</span>\r\n              </a>\r\n              <meta itemprop=\"position\" content=\"1\" />\r\n              <span class=\"ui-page-header__separator\" aria-hidden=\"true\">\r\n                <lucide-icon name=\"chevron-right\" [size]=\"14\" />\r\n              </span>\r\n            </li>\r\n          }\r\n          @for (crumb of breadcrumbs; track crumb.url; let i = $index) {\r\n            <li\r\n              class=\"ui-page-header__item\"\r\n              [class.ui-page-header__item--current]=\"crumb.isLast\"\r\n              itemprop=\"itemListElement\"\r\n              itemscope\r\n              itemtype=\"https://schema.org/ListItem\"\r\n            >\r\n              @if (!crumb.isLast) {\r\n                <a\r\n                  itemprop=\"item\"\r\n                  [routerLink]=\"crumb.url\"\r\n                  class=\"ui-page-header__link\"\r\n                >\r\n                  <span itemprop=\"name\" class=\"ui-page-header__text\">{{ crumb.label }}</span>\r\n                </a>\r\n                <span class=\"ui-page-header__separator\" aria-hidden=\"true\">\r\n                  <lucide-icon name=\"chevron-right\" [size]=\"14\" />\r\n                </span>\r\n              } @else {\r\n                <span\r\n                  itemprop=\"name\"\r\n                  class=\"ui-page-header__text ui-page-header__text--current\"\r\n                  aria-current=\"page\"\r\n                >\r\n                  {{ crumb.label }}\r\n                </span>\r\n              }\r\n              <meta itemprop=\"position\" [attr.content]=\"showHome ? i + 2 : i + 1\" />\r\n            </li>\r\n          }\r\n        </ol>\r\n      </nav>\r\n\r\n      <h1 class=\"ui-page-header__title\" id=\"page-title\">\r\n        {{ displayTitle }}\r\n      </h1>\r\n    </header>\r\n  `,\r\n  styleUrl: './page-header.component.scss',\r\n})\r\nexport class UiPageHeaderComponent implements OnInit {\r\n  /** Override the auto-detected page title. When empty, title is derived from the last breadcrumb. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link at the start of the trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on each navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title` (e.g. `' | My App'`). Only used when `updateDocumentTitle` is `true`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** @internal */\r\n  breadcrumbs: UiBreadcrumbItem[] = [];\r\n\r\n  private readonly router = inject(Router);\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly titleService = inject(Title);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n\r\n  /** Resolved page title: explicit `title` input wins, otherwise last breadcrumb label. */\r\n  get displayTitle(): string {\r\n    if (this.title) return this.title;\r\n    if (this.breadcrumbs.length > 0) {\r\n      return this.breadcrumbs[this.breadcrumbs.length - 1].label;\r\n    }\r\n    return '';\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.refresh();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.refresh());\r\n  }\r\n\r\n  /**\r\n   * Navigates to the previous breadcrumb (parent route).\r\n   * @param fallbackUrl - URL to navigate to when no parent exists (default: homeRoute).\r\n   */\r\n  navigateBack(fallbackUrl?: string): void {\r\n    this.breadcrumbService.navigateBack(fallbackUrl ?? this.homeRoute);\r\n  }\r\n\r\n  /** @internal */\r\n  private refresh(): void {\r\n    this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n    if (this.updateDocumentTitle) {\r\n      const title = this.displayTitle;\r\n      if (title) {\r\n        this.titleService.setTitle(`${title}${this.titleSuffix}`);\r\n      }\r\n    }\r\n    this.cdr.markForCheck();\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,21 @@
1
+ /**
2
+ * @module ng-ui-system/page-header
3
+ * Types and interfaces for UiPageHeader and UiBreadcrumbService.
4
+ */
5
+ import { InjectionToken } from '@angular/core';
6
+ /**
7
+ * Injection token for providing a custom label resolver to UiBreadcrumbService.
8
+ *
9
+ * When not provided, parameters are displayed as-is (identity function).
10
+ *
11
+ * @example
12
+ * ```typescript
13
+ * // In your app.config.ts or module providers:
14
+ * { provide: UI_BREADCRUMB_LABEL_RESOLVER, useValue: myLabelResolver }
15
+ * ```
16
+ */
17
+ export const UI_BREADCRUMB_LABEL_RESOLVER = new InjectionToken('UI_BREADCRUMB_LABEL_RESOLVER', {
18
+ providedIn: 'root',
19
+ factory: () => (_name, value) => value,
20
+ });
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS1oZWFkZXIudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL3BhZ2UtaGVhZGVyL3BhZ2UtaGVhZGVyLnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFvRi9DOzs7Ozs7Ozs7O0dBVUc7QUFDSCxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxJQUFJLGNBQWMsQ0FBb0IsOEJBQThCLEVBQUU7SUFDaEgsVUFBVSxFQUFFLE1BQU07SUFDbEIsT0FBTyxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUMsS0FBYSxFQUFFLEtBQWEsRUFBRSxFQUFFLENBQUMsS0FBSztDQUN2RCxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogQG1vZHVsZSBuZy11aS1zeXN0ZW0vcGFnZS1oZWFkZXJcclxuICogVHlwZXMgYW5kIGludGVyZmFjZXMgZm9yIFVpUGFnZUhlYWRlciBhbmQgVWlCcmVhZGNydW1iU2VydmljZS5cclxuICovXHJcblxyXG5pbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuLyoqXHJcbiAqIFNpbmdsZSBicmVhZGNydW1iIGl0ZW0gcmVuZGVyZWQgaW4gdGhlIG5hdmlnYXRpb24gdHJhaWwuXHJcbiAqXHJcbiAqIEB1c2FnZU5vdGVzXHJcbiAqICMjIyBTdHJ1Y3R1cmVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBjb25zdCBpdGVtOiBVaUJyZWFkY3J1bWJJdGVtID0ge1xyXG4gKiAgIGxhYmVsOiAnRGFzaGJvYXJkJyxcclxuICogICB1cmw6ICcvZGFzaGJvYXJkJyxcclxuICogICBpc0xhc3Q6IGZhbHNlLFxyXG4gKiB9O1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWlCcmVhZGNydW1iSXRlbSB7XHJcbiAgLyoqIERpc3BsYXkgdGV4dCBmb3IgdGhlIGJyZWFkY3J1bWIgbGluay4gKi9cclxuICBsYWJlbDogc3RyaW5nO1xyXG4gIC8qKiBOYXZpZ2F0aW9uIFVSTCAoQW5ndWxhciBSb3V0ZXIgcGF0aCkuICovXHJcbiAgdXJsOiBzdHJpbmc7XHJcbiAgLyoqIFdoZXRoZXIgdGhpcyBpcyB0aGUgbGFzdCAoY3VycmVudCkgYnJlYWRjcnVtYiBpbiB0aGUgdHJhaWwuICovXHJcbiAgaXNMYXN0OiBib29sZWFuO1xyXG59XHJcblxyXG4vKipcclxuICogSW50ZXJuYWwgcm91dGUgY29uZmlndXJhdGlvbiBzdG9yZWQgYnkgVWlCcmVhZGNydW1iU2VydmljZS5cclxuICogQnVpbHQgZnJvbSBBbmd1bGFyIHJvdXRlIGBkYXRhYCBwcm9wZXJ0aWVzLlxyXG4gKlxyXG4gKiBAdXNhZ2VOb3Rlc1xyXG4gKiAjIyMgUm91dGUgZGF0YSBjb252ZW50aW9uXHJcbiAqIGBgYHR5cGVzY3JpcHRcclxuICoge1xyXG4gKiAgIHBhdGg6ICd1c2Vycy86dXNlcklkJyxcclxuICogICBkYXRhOiB7XHJcbiAqICAgICBpZDogJ3VzZXItZGV0YWlsJyxcclxuICogICAgIHRpdGxlOiAnRGV0dGFnbGlvIHV0ZW50ZScsXHJcbiAqICAgICBicmVhZGNydW1iczogWyd1c2Vycy1saXN0J10sXHJcbiAqICAgfSxcclxuICogICBjb21wb25lbnQ6IFVzZXJEZXRhaWxDb21wb25lbnQsXHJcbiAqIH1cclxuICogYGBgXHJcbiAqL1xyXG5leHBvcnQgaW50ZXJmYWNlIFVpQnJlYWRjcnVtYkNvbmZpZyB7XHJcbiAgLyoqIFVuaXF1ZSByb3V0ZSBpZGVudGlmaWVyIChmcm9tIGByb3V0ZS5kYXRhLmlkYCkuICovXHJcbiAgaWQ6IHN0cmluZztcclxuICAvKiogUmVzb2x2ZWQgcm91dGUgcGF0aCBwYXR0ZXJuIChtYXkgY29udGFpbiBgOnBhcmFtYCBwbGFjZWhvbGRlcnMpLiAqL1xyXG4gIHBhdGg6IHN0cmluZztcclxuICAvKiogRGlzcGxheSBsYWJlbCAoZnJvbSBgcm91dGUuZGF0YS50aXRsZWAgb3IgZmFsbGJhY2spLiAqL1xyXG4gIGxhYmVsOiBzdHJpbmc7XHJcbiAgLyoqIE9yZGVyZWQgYXJyYXkgb2YgcGFyZW50IHJvdXRlIElEcyBmb3IgYnVpbGRpbmcgdGhlIGJyZWFkY3J1bWIgY2hhaW4uICovXHJcbiAgcGFyZW50SWRzOiBzdHJpbmdbXTtcclxuICAvKiogRnVsbHkgcmVzb2x2ZWQgVVJMIGNhcHR1cmVkIGF0IG5hdmlnYXRpb24gdGltZS4gKi9cclxuICBmdWxsUGF0aD86IHN0cmluZztcclxuICAvKiogUmF3IHJvdXRlIGRhdGEgbWV0YWRhdGEuICovXHJcbiAgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xyXG59XHJcblxyXG4vKipcclxuICogRnVuY3Rpb24gc2lnbmF0dXJlIGZvciByZXNvbHZpbmcgcm91dGUgcGFyYW1ldGVyIHZhbHVlcyB0byBodW1hbi1yZWFkYWJsZSBsYWJlbHMuXHJcbiAqXHJcbiAqIENvbnN1bWVycyBjYW4gcHJvdmlkZSBhIGN1c3RvbSBpbXBsZW1lbnRhdGlvbiB2aWEgYFVJX0JSRUFEQ1JVTUJfTEFCRUxfUkVTT0xWRVJgXHJcbiAqIHRvIHRyYW5zbGF0ZSBjcnlwdGljIHBhcmFtZXRlciB2YWx1ZXMgKGUuZy4gVVVJRHMsIHNsdWdzKSBpbnRvIHVzZXItZnJpZW5kbHkgdGV4dC5cclxuICpcclxuICogQHBhcmFtIHBhcmFtTmFtZSAtIFRoZSByb3V0ZSBwYXJhbWV0ZXIgbmFtZSAoZS5nLiBgJ3VzZXJJZCdgKS5cclxuICogQHBhcmFtIHBhcmFtVmFsdWUgLSBUaGUgcmF3IHBhcmFtZXRlciB2YWx1ZSBmcm9tIHRoZSBVUkwgKGUuZy4gYCdhYmMtMTIzJ2ApLlxyXG4gKiBAcmV0dXJucyBBIGh1bWFuLXJlYWRhYmxlIGxhYmVsLCBvciB0aGUgb3JpZ2luYWwgdmFsdWUgaWYgbm8gbWFwcGluZyBleGlzdHMuXHJcbiAqXHJcbiAqIEB1c2FnZU5vdGVzXHJcbiAqICMjIyBDdXN0b20gcmVzb2x2ZXJcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBjb25zdCBteVJlc29sdmVyOiBVaUxhYmVsUmVzb2x2ZXJGbiA9IChuYW1lLCB2YWx1ZSkgPT4ge1xyXG4gKiAgIGlmIChuYW1lID09PSAnc3RhdHVzJykge1xyXG4gKiAgICAgcmV0dXJuIHsgcGVuZGluZzogJ0luIGF0dGVzYScsIGFjdGl2ZTogJ0F0dGl2bycgfVt2YWx1ZV0gPz8gdmFsdWU7XHJcbiAqICAgfVxyXG4gKiAgIHJldHVybiB2YWx1ZTtcclxuICogfTtcclxuICpcclxuICogcHJvdmlkZXJzOiBbXHJcbiAqICAgeyBwcm92aWRlOiBVSV9CUkVBRENSVU1CX0xBQkVMX1JFU09MVkVSLCB1c2VWYWx1ZTogbXlSZXNvbHZlciB9LFxyXG4gKiBdXHJcbiAqIGBgYFxyXG4gKi9cclxuZXhwb3J0IHR5cGUgVWlMYWJlbFJlc29sdmVyRm4gPSAocGFyYW1OYW1lOiBzdHJpbmcsIHBhcmFtVmFsdWU6IHN0cmluZykgPT4gc3RyaW5nO1xyXG5cclxuLyoqXHJcbiAqIEluamVjdGlvbiB0b2tlbiBmb3IgcHJvdmlkaW5nIGEgY3VzdG9tIGxhYmVsIHJlc29sdmVyIHRvIFVpQnJlYWRjcnVtYlNlcnZpY2UuXHJcbiAqXHJcbiAqIFdoZW4gbm90IHByb3ZpZGVkLCBwYXJhbWV0ZXJzIGFyZSBkaXNwbGF5ZWQgYXMtaXMgKGlkZW50aXR5IGZ1bmN0aW9uKS5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiAvLyBJbiB5b3VyIGFwcC5jb25maWcudHMgb3IgbW9kdWxlIHByb3ZpZGVyczpcclxuICogeyBwcm92aWRlOiBVSV9CUkVBRENSVU1CX0xBQkVMX1JFU09MVkVSLCB1c2VWYWx1ZTogbXlMYWJlbFJlc29sdmVyIH1cclxuICogYGBgXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgVUlfQlJFQURDUlVNQl9MQUJFTF9SRVNPTFZFUiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxVaUxhYmVsUmVzb2x2ZXJGbj4oJ1VJX0JSRUFEQ1JVTUJfTEFCRUxfUkVTT0xWRVInLCB7XHJcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxyXG4gIGZhY3Rvcnk6ICgpID0+IChfbmFtZTogc3RyaW5nLCB2YWx1ZTogc3RyaW5nKSA9PiB2YWx1ZSxcclxufSk7XHJcbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export { UiPaginatedTableComponent } from './paginated-table.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL3RhYmxlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgVWlQYWdpbmF0ZWRUYWJsZUNvbXBvbmVudCB9IGZyb20gJy4vcGFnaW5hdGVkLXRhYmxlLmNvbXBvbmVudCc7XHJcbmV4cG9ydCB7XHJcbiAgVWlUYWJsZUNvbHVtbixcclxuICBVaVRhYmxlU29ydEV2ZW50LFxyXG4gIFVpVGFibGVQYWdlRXZlbnQsXHJcbn0gZnJvbSAnLi90YWJsZS50eXBlcyc7XHJcbiJdfQ==