@gnggln/ng-ui-system 1.0.0-alpha.7 → 1.0.0-alpha.8
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.
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +25 -26
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +5 -4
- package/fesm2022/gnggln-ng-ui-system.mjs +24 -23
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -1
- package/lib/components/base-layout/base-layout.component.d.ts +7 -10
- package/lib/components/page-header/breadcrumb.service.d.ts +2 -2
- package/package.json +1 -1
|
@@ -1,11 +1,10 @@
|
|
|
1
|
-
import { Component, DestroyRef,
|
|
1
|
+
import { ChangeDetectionStrategy, Component, DestroyRef, Input, inject, ViewEncapsulation, } from '@angular/core';
|
|
2
2
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
3
|
-
import { Location } from '@angular/common';
|
|
4
3
|
import { ActivatedRoute, NavigationEnd, Router } from '@angular/router';
|
|
5
|
-
import {
|
|
6
|
-
import { UiPageHeaderComponent } from '../page-header/page-header.component';
|
|
7
|
-
import { UiBreadcrumbService } from '../page-header/breadcrumb.service';
|
|
4
|
+
import { debounceTime, filter } from 'rxjs/operators';
|
|
8
5
|
import { UiButtonAreaComponent } from '../button/button-area.component';
|
|
6
|
+
import { UiBreadcrumbService } from '../page-header/breadcrumb.service';
|
|
7
|
+
import { UiPageHeaderComponent } from '../page-header/page-header.component';
|
|
9
8
|
import * as i0 from "@angular/core";
|
|
10
9
|
/**
|
|
11
10
|
* Full-page layout component that combines a page header (breadcrumbs + title),
|
|
@@ -63,12 +62,9 @@ export class UiBaseLayoutComponent {
|
|
|
63
62
|
/** Gap size between footer buttons. */
|
|
64
63
|
this.footerGap = 'sm';
|
|
65
64
|
this.router = inject(Router);
|
|
66
|
-
this.location = inject(Location);
|
|
67
65
|
this.activatedRoute = inject(ActivatedRoute);
|
|
68
66
|
this.breadcrumbService = inject(UiBreadcrumbService);
|
|
69
67
|
this.destroyRef = inject(DestroyRef);
|
|
70
|
-
/** @internal Current breadcrumbs, updated on each navigation. */
|
|
71
|
-
this.breadcrumbs = [];
|
|
72
68
|
}
|
|
73
69
|
/**
|
|
74
70
|
* Merged action list (back button + consumer actions).
|
|
@@ -78,14 +74,13 @@ export class UiBaseLayoutComponent {
|
|
|
78
74
|
get mergedActions() {
|
|
79
75
|
const result = [];
|
|
80
76
|
if (this.showBackButton) {
|
|
81
|
-
const backTarget = this.getBackTarget();
|
|
82
77
|
result.push({
|
|
83
78
|
id: '__ui-back',
|
|
84
79
|
label: this.backButtonConfig.label ?? 'Indietro',
|
|
85
80
|
icon: this.backButtonConfig.icon ?? 'arrow-left',
|
|
86
81
|
iconPosition: 'leading',
|
|
87
82
|
variant: this.backButtonConfig.variant ?? 'outline',
|
|
88
|
-
hidden:
|
|
83
|
+
hidden: this.shouldHideBackButton(),
|
|
89
84
|
action: () => this.navigateBack(),
|
|
90
85
|
});
|
|
91
86
|
}
|
|
@@ -102,25 +97,29 @@ export class UiBaseLayoutComponent {
|
|
|
102
97
|
.pipe(filter((event) => event instanceof NavigationEnd), debounceTime(50), takeUntilDestroyed(this.destroyRef))
|
|
103
98
|
.subscribe(() => this.refreshBreadcrumbs());
|
|
104
99
|
}
|
|
105
|
-
/** @internal
|
|
100
|
+
/** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */
|
|
106
101
|
refreshBreadcrumbs() {
|
|
107
|
-
this.
|
|
108
|
-
}
|
|
109
|
-
/** @internal Determines the back navigation target from breadcrumbs. */
|
|
110
|
-
getBackTarget() {
|
|
111
|
-
if (this.breadcrumbs.length < 2)
|
|
112
|
-
return null;
|
|
113
|
-
return this.breadcrumbs[this.breadcrumbs.length - 2];
|
|
102
|
+
this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
|
|
114
103
|
}
|
|
115
|
-
/** @internal
|
|
116
|
-
|
|
117
|
-
const
|
|
118
|
-
if (
|
|
119
|
-
|
|
104
|
+
/** @internal Determina se il back button deve essere nascosto per la rotta corrente. */
|
|
105
|
+
shouldHideBackButton() {
|
|
106
|
+
const currentPath = this.router.url.split('?')[0].split('#')[0];
|
|
107
|
+
if (currentPath === this.homeRoute) {
|
|
108
|
+
return true;
|
|
109
|
+
}
|
|
110
|
+
let deepestRoute = this.activatedRoute;
|
|
111
|
+
while (deepestRoute.firstChild) {
|
|
112
|
+
deepestRoute = deepestRoute.firstChild;
|
|
120
113
|
}
|
|
121
|
-
|
|
122
|
-
|
|
114
|
+
// biome-ignore lint/complexity/useLiteralKeys: <explanation>
|
|
115
|
+
if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {
|
|
116
|
+
return true;
|
|
123
117
|
}
|
|
118
|
+
return false;
|
|
119
|
+
}
|
|
120
|
+
/** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */
|
|
121
|
+
navigateBack() {
|
|
122
|
+
this.breadcrumbService.navigateBack(this.homeRoute);
|
|
124
123
|
}
|
|
125
124
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
126
125
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiBaseLayoutComponent, isStandalone: true, selector: "ui-base-layout", inputs: { showHeader: "showHeader", title: "title", homeRoute: "homeRoute", showHome: "showHome", updateDocumentTitle: "updateDocumentTitle", titleSuffix: "titleSuffix", showBackButton: "showBackButton", backButtonConfig: "backButtonConfig", actions: "actions", footerAlign: "footerAlign", footerGap: "footerGap" }, host: { classAttribute: "ui-base-layout-host" }, ngImport: i0, template: `
|
|
@@ -215,4 +214,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
215
214
|
}], footerGap: [{
|
|
216
215
|
type: Input
|
|
217
216
|
}] } });
|
|
218
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-layout.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/base-layout/base-layout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  DestroyRef,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  ChangeDetectionStrategy,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { Location } from '@angular/common';\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { filter, debounceTime } from 'rxjs/operators';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiBreadcrumbItem } from '../page-header/page-header.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiButtonDescriptor, UiButtonAreaAlign } from '../button/button.types';\r\nimport { UiBackButtonConfig } from './base-layout.types';\r\n\r\n/**\r\n * Full-page layout component that combines a page header (breadcrumbs + title),\r\n * a main content area via `<ng-content>`, and a sticky footer action bar\r\n * powered by `UiButtonAreaComponent`.\r\n *\r\n * Optionally renders an automatic \"back\" button in the footer, derived from\r\n * the breadcrumb trail (navigates to the parent breadcrumb).\r\n *\r\n * @selector ui-base-layout\r\n *\r\n * @example\r\n * ```html\r\n * <ui-base-layout [actions]=\"pageActions\" footerAlign=\"end\">\r\n *   <p>Your page content here</p>\r\n * </ui-base-layout>\r\n * ```\r\n *\r\n * @example\r\n * ```html\r\n * <!-- With back button and custom title -->\r\n * <ui-base-layout\r\n *   title=\"Dettaglio utente\"\r\n *   [showBackButton]=\"true\"\r\n *   [actions]=\"[\r\n *     { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },\r\n *   ]\"\r\n * >\r\n *   <app-user-detail />\r\n * </ui-base-layout>\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-base-layout',\r\n  standalone: true,\r\n  imports: [UiPageHeaderComponent, UiButtonAreaComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-base-layout-host',\r\n  },\r\n  template: `\r\n    <div class=\"ui-base-layout\">\r\n      @if (showHeader) {\r\n        <ui-page-header\r\n          [title]=\"title\"\r\n          [homeRoute]=\"homeRoute\"\r\n          [showHome]=\"showHome\"\r\n          [updateDocumentTitle]=\"updateDocumentTitle\"\r\n          [titleSuffix]=\"titleSuffix\"\r\n        />\r\n      }\r\n\r\n      <main class=\"ui-base-layout__content\">\r\n        <ng-content />\r\n      </main>\r\n\r\n      @if (hasVisibleActions) {\r\n        <aside\r\n          class=\"ui-base-layout__footer\"\r\n          aria-label=\"Page actions\"\r\n        >\r\n          <ui-button-area\r\n            [buttons]=\"mergedActions\"\r\n            [align]=\"footerAlign\"\r\n            [gap]=\"footerGap\"\r\n            [stackOnMobile]=\"true\"\r\n            ariaLabel=\"Page actions\"\r\n          />\r\n        </aside>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './base-layout.component.scss',\r\n})\r\nexport class UiBaseLayoutComponent implements OnInit {\r\n  /** Whether to display the page header (breadcrumbs + title). */\r\n  @Input() showHeader = true;\r\n\r\n  /** Override the auto-detected page title. Passed through to `ui-page-header`. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link in the breadcrumb trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** Whether to show an automatic \"back\" button based on breadcrumbs. */\r\n  @Input() showBackButton = true;\r\n\r\n  /** Configuration overrides for the automatic back button. */\r\n  @Input() backButtonConfig: UiBackButtonConfig = {};\r\n\r\n  /** Array of button descriptors for the footer action area. */\r\n  @Input() actions: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the footer button area. */\r\n  @Input() footerAlign: UiButtonAreaAlign = 'start';\r\n\r\n  /** Gap size between footer buttons. */\r\n  @Input() footerGap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'sm';\r\n\r\n  private readonly router = inject(Router);\r\n  private readonly location = inject(Location);\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n\r\n  /** @internal Current breadcrumbs, updated on each navigation. */\r\n  private breadcrumbs: UiBreadcrumbItem[] = [];\r\n\r\n  /**\r\n   * Merged action list (back button + consumer actions).\r\n   * Computed on every change-detection check so it stays in sync\r\n   * with both navigation state (breadcrumbs) and input changes.\r\n   */\r\n  get mergedActions(): UiButtonDescriptor[] {\r\n    const result: UiButtonDescriptor[] = [];\r\n\r\n    if (this.showBackButton) {\r\n      const backTarget = this.getBackTarget();\r\n      result.push({\r\n        id: '__ui-back',\r\n        label: this.backButtonConfig.label ?? 'Indietro',\r\n        icon: (this.backButtonConfig.icon as any) ?? 'arrow-left',\r\n        iconPosition: 'leading',\r\n        variant: (this.backButtonConfig.variant as any) ?? 'outline',\r\n        hidden: !backTarget,\r\n        action: () => this.navigateBack(),\r\n      });\r\n    }\r\n\r\n    result.push(...this.actions);\r\n    return result;\r\n  }\r\n\r\n  /** Whether the footer should render (at least one non-hidden action exists). */\r\n  get hasVisibleActions(): boolean {\r\n    return this.mergedActions.some((a) => !a.hidden);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.refreshBreadcrumbs();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        debounceTime(50),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.refreshBreadcrumbs());\r\n  }\r\n\r\n  /** @internal Refreshes breadcrumb state from the current route. */\r\n  private refreshBreadcrumbs(): void {\r\n    this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Determines the back navigation target from breadcrumbs. */\r\n  private getBackTarget(): UiBreadcrumbItem | null {\r\n    if (this.breadcrumbs.length < 2) return null;\r\n    return this.breadcrumbs[this.breadcrumbs.length - 2];\r\n  }\r\n\r\n  /** @internal Navigates to the parent breadcrumb or falls back to browser history. */\r\n  private navigateBack(): void {\r\n    const target = this.getBackTarget();\r\n    if (target?.url) {\r\n      this.router.navigateByUrl(target.url);\r\n    } else {\r\n      this.location.back();\r\n    }\r\n  }\r\n}\r\n"]}
|
|
217
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-layout.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/base-layout/base-layout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  DestroyRef,\r\n  Input,\r\n  inject,\r\n  type OnInit,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\n\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { debounceTime, filter } from 'rxjs/operators';\r\nimport type { UiButtonAreaAlign, UiButtonDescriptor } from '../button/button.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport type { UiBackButtonConfig } from './base-layout.types';\r\n\r\n/**\r\n * Full-page layout component that combines a page header (breadcrumbs + title),\r\n * a main content area via `<ng-content>`, and a sticky footer action bar\r\n * powered by `UiButtonAreaComponent`.\r\n *\r\n * Optionally renders an automatic \"back\" button in the footer, derived from\r\n * the breadcrumb trail (navigates to the parent breadcrumb).\r\n *\r\n * @selector ui-base-layout\r\n *\r\n * @example\r\n * ```html\r\n * <ui-base-layout [actions]=\"pageActions\" footerAlign=\"end\">\r\n *   <p>Your page content here</p>\r\n * </ui-base-layout>\r\n * ```\r\n *\r\n * @example\r\n * ```html\r\n * <!-- With back button and custom title -->\r\n * <ui-base-layout\r\n *   title=\"Dettaglio utente\"\r\n *   [showBackButton]=\"true\"\r\n *   [actions]=\"[\r\n *     { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },\r\n *   ]\"\r\n * >\r\n *   <app-user-detail />\r\n * </ui-base-layout>\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-base-layout',\r\n  standalone: true,\r\n  imports: [UiPageHeaderComponent, UiButtonAreaComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-base-layout-host',\r\n  },\r\n  template: `\r\n    <div class=\"ui-base-layout\">\r\n      @if (showHeader) {\r\n        <ui-page-header\r\n          [title]=\"title\"\r\n          [homeRoute]=\"homeRoute\"\r\n          [showHome]=\"showHome\"\r\n          [updateDocumentTitle]=\"updateDocumentTitle\"\r\n          [titleSuffix]=\"titleSuffix\"\r\n        />\r\n      }\r\n\r\n      <main class=\"ui-base-layout__content\">\r\n        <ng-content />\r\n      </main>\r\n\r\n      @if (hasVisibleActions) {\r\n        <aside\r\n          class=\"ui-base-layout__footer\"\r\n          aria-label=\"Page actions\"\r\n        >\r\n          <ui-button-area\r\n            [buttons]=\"mergedActions\"\r\n            [align]=\"footerAlign\"\r\n            [gap]=\"footerGap\"\r\n            [stackOnMobile]=\"true\"\r\n            ariaLabel=\"Page actions\"\r\n          />\r\n        </aside>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './base-layout.component.scss',\r\n})\r\nexport class UiBaseLayoutComponent implements OnInit {\r\n  /** Whether to display the page header (breadcrumbs + title). */\r\n  @Input() showHeader = true;\r\n\r\n  /** Override the auto-detected page title. Passed through to `ui-page-header`. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link in the breadcrumb trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** Whether to show an automatic \"back\" button based on breadcrumbs. */\r\n  @Input() showBackButton = true;\r\n\r\n  /** Configuration overrides for the automatic back button. */\r\n  @Input() backButtonConfig: UiBackButtonConfig = {};\r\n\r\n  /** Array of button descriptors for the footer action area. */\r\n  @Input() actions: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the footer button area. */\r\n  @Input() footerAlign: UiButtonAreaAlign = 'start';\r\n\r\n  /** Gap size between footer buttons. */\r\n  @Input() footerGap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'sm';\r\n\r\n  private readonly router = inject(Router);\r\n\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n\r\n\r\n\r\n  /**\r\n   * Merged action list (back button + consumer actions).\r\n   * Computed on every change-detection check so it stays in sync\r\n   * with both navigation state (breadcrumbs) and input changes.\r\n   */\r\n  get mergedActions(): UiButtonDescriptor[] {\r\n    const result: UiButtonDescriptor[] = [];\r\n\r\n    if (this.showBackButton) {\r\n      result.push({\r\n        id: '__ui-back',\r\n        label: this.backButtonConfig.label ?? 'Indietro',\r\n        icon: (this.backButtonConfig.icon as any) ?? 'arrow-left',\r\n        iconPosition: 'leading',\r\n        variant: (this.backButtonConfig.variant as any) ?? 'outline',\r\n        hidden: this.shouldHideBackButton(),\r\n        action: () => this.navigateBack(),\r\n      });\r\n    }\r\n\r\n    result.push(...this.actions);\r\n    return result;\r\n  }\r\n\r\n  /** Whether the footer should render (at least one non-hidden action exists). */\r\n  get hasVisibleActions(): boolean {\r\n    return this.mergedActions.some((a) => !a.hidden);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.refreshBreadcrumbs();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        debounceTime(50),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.refreshBreadcrumbs());\r\n  }\r\n\r\n  /** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */\r\n  private refreshBreadcrumbs(): void {\r\n    this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Determina se il back button deve essere nascosto per la rotta corrente. */\r\n  private shouldHideBackButton(): boolean {\r\n    const currentPath = this.router.url.split('?')[0].split('#')[0];\r\n    if (currentPath === this.homeRoute) {\r\n      return true;\r\n    }\r\n\r\n    let deepestRoute = this.activatedRoute;\r\n    while (deepestRoute.firstChild) {\r\n      deepestRoute = deepestRoute.firstChild;\r\n    }\r\n\r\n    // biome-ignore lint/complexity/useLiteralKeys: <explanation>\r\n    if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {\r\n      return true;\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  /** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */\r\n  private navigateBack(): void {\r\n    this.breadcrumbService.navigateBack(this.homeRoute);\r\n  }\r\n}\r\n"]}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
|
|
1
|
+
/** biome-ignore-all lint/complexity/useLiteralKeys: <explanation> */
|
|
2
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
3
|
import { NavigationEnd, Router } from '@angular/router';
|
|
3
4
|
import { filter } from 'rxjs/operators';
|
|
4
5
|
import { UI_BREADCRUMB_LABEL_RESOLVER, } from './page-header.types';
|
|
@@ -79,7 +80,7 @@ export class UiBreadcrumbService {
|
|
|
79
80
|
this.lastBreadcrumbs = [];
|
|
80
81
|
return [];
|
|
81
82
|
}
|
|
82
|
-
const params = this.collectParamsFromPath(
|
|
83
|
+
const params = this.collectParamsFromPath(deepestRoute);
|
|
83
84
|
const breadcrumbs = this.buildBreadcrumbChain(routeId, params);
|
|
84
85
|
this.lastBreadcrumbs = breadcrumbs;
|
|
85
86
|
return breadcrumbs;
|
|
@@ -131,7 +132,7 @@ export class UiBreadcrumbService {
|
|
|
131
132
|
/** @internal Recursively walks route tree to build the breadcrumb map. */
|
|
132
133
|
buildRouteMap(routes, parentPath = '') {
|
|
133
134
|
for (const route of routes) {
|
|
134
|
-
const currentPath = parentPath
|
|
135
|
+
const currentPath = `${parentPath}/${route.path || ''}`;
|
|
135
136
|
if (route.data?.['id']) {
|
|
136
137
|
const fullPath = this.normalizePath(currentPath);
|
|
137
138
|
this.breadcrumbMap.set(route.data['id'], {
|
|
@@ -239,4 +240,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
239
240
|
type: Injectable,
|
|
240
241
|
args: [{ providedIn: 'root' }]
|
|
241
242
|
}], 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"]}
|
|
243
|
+
//# 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":["/** biome-ignore-all lint/complexity/useLiteralKeys: <explanation> */\r\nimport { Injectable, inject } from '@angular/core';\r\nimport { type ActivatedRoute, NavigationEnd, Router, type Routes } from '@angular/router';\r\nimport { filter } from 'rxjs/operators';\r\nimport {\r\n  UI_BREADCRUMB_LABEL_RESOLVER,\r\n  type UiBreadcrumbConfig,\r\n  type UiBreadcrumbItem,\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(deepestRoute);\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"]}
|
|
@@ -6,7 +6,7 @@ import * as i1$1 from 'lucide-angular';
|
|
|
6
6
|
import { LucideAngularModule } from 'lucide-angular';
|
|
7
7
|
import { Router, NavigationEnd, ActivatedRoute, RouterLink, RouterOutlet, RouterLinkActive } from '@angular/router';
|
|
8
8
|
import * as i2$3 from '@angular/common';
|
|
9
|
-
import { isPlatformBrowser, NgTemplateOutlet,
|
|
9
|
+
import { isPlatformBrowser, NgTemplateOutlet, CommonModule, DOCUMENT } from '@angular/common';
|
|
10
10
|
import { Subject, Observable, map, combineLatest, startWith, distinctUntilChanged, takeUntil as takeUntil$1, BehaviorSubject, interval, filter as filter$1, throwError } from 'rxjs';
|
|
11
11
|
import { filter, takeUntil, debounceTime, distinctUntilChanged as distinctUntilChanged$1, tap, catchError, map as map$1, finalize } from 'rxjs/operators';
|
|
12
12
|
import * as i1$6 from '@angular/material/dialog';
|
|
@@ -49094,6 +49094,7 @@ const UI_BREADCRUMB_LABEL_RESOLVER = new InjectionToken('UI_BREADCRUMB_LABEL_RES
|
|
|
49094
49094
|
factory: () => (_name, value) => value,
|
|
49095
49095
|
});
|
|
49096
49096
|
|
|
49097
|
+
/** biome-ignore-all lint/complexity/useLiteralKeys: <explanation> */
|
|
49097
49098
|
/**
|
|
49098
49099
|
* Service that builds a nested breadcrumb trail from Angular route configuration.
|
|
49099
49100
|
*
|
|
@@ -49170,7 +49171,7 @@ class UiBreadcrumbService {
|
|
|
49170
49171
|
this.lastBreadcrumbs = [];
|
|
49171
49172
|
return [];
|
|
49172
49173
|
}
|
|
49173
|
-
const params = this.collectParamsFromPath(
|
|
49174
|
+
const params = this.collectParamsFromPath(deepestRoute);
|
|
49174
49175
|
const breadcrumbs = this.buildBreadcrumbChain(routeId, params);
|
|
49175
49176
|
this.lastBreadcrumbs = breadcrumbs;
|
|
49176
49177
|
return breadcrumbs;
|
|
@@ -49222,7 +49223,7 @@ class UiBreadcrumbService {
|
|
|
49222
49223
|
/** @internal Recursively walks route tree to build the breadcrumb map. */
|
|
49223
49224
|
buildRouteMap(routes, parentPath = '') {
|
|
49224
49225
|
for (const route of routes) {
|
|
49225
|
-
const currentPath = parentPath
|
|
49226
|
+
const currentPath = `${parentPath}/${route.path || ''}`;
|
|
49226
49227
|
if (route.data?.['id']) {
|
|
49227
49228
|
const fullPath = this.normalizePath(currentPath);
|
|
49228
49229
|
this.breadcrumbMap.set(route.data['id'], {
|
|
@@ -49636,12 +49637,9 @@ class UiBaseLayoutComponent {
|
|
|
49636
49637
|
/** Gap size between footer buttons. */
|
|
49637
49638
|
this.footerGap = 'sm';
|
|
49638
49639
|
this.router = inject(Router);
|
|
49639
|
-
this.location = inject(Location);
|
|
49640
49640
|
this.activatedRoute = inject(ActivatedRoute);
|
|
49641
49641
|
this.breadcrumbService = inject(UiBreadcrumbService);
|
|
49642
49642
|
this.destroyRef = inject(DestroyRef);
|
|
49643
|
-
/** @internal Current breadcrumbs, updated on each navigation. */
|
|
49644
|
-
this.breadcrumbs = [];
|
|
49645
49643
|
}
|
|
49646
49644
|
/**
|
|
49647
49645
|
* Merged action list (back button + consumer actions).
|
|
@@ -49651,14 +49649,13 @@ class UiBaseLayoutComponent {
|
|
|
49651
49649
|
get mergedActions() {
|
|
49652
49650
|
const result = [];
|
|
49653
49651
|
if (this.showBackButton) {
|
|
49654
|
-
const backTarget = this.getBackTarget();
|
|
49655
49652
|
result.push({
|
|
49656
49653
|
id: '__ui-back',
|
|
49657
49654
|
label: this.backButtonConfig.label ?? 'Indietro',
|
|
49658
49655
|
icon: this.backButtonConfig.icon ?? 'arrow-left',
|
|
49659
49656
|
iconPosition: 'leading',
|
|
49660
49657
|
variant: this.backButtonConfig.variant ?? 'outline',
|
|
49661
|
-
hidden:
|
|
49658
|
+
hidden: this.shouldHideBackButton(),
|
|
49662
49659
|
action: () => this.navigateBack(),
|
|
49663
49660
|
});
|
|
49664
49661
|
}
|
|
@@ -49675,25 +49672,29 @@ class UiBaseLayoutComponent {
|
|
|
49675
49672
|
.pipe(filter((event) => event instanceof NavigationEnd), debounceTime(50), takeUntilDestroyed(this.destroyRef))
|
|
49676
49673
|
.subscribe(() => this.refreshBreadcrumbs());
|
|
49677
49674
|
}
|
|
49678
|
-
/** @internal
|
|
49675
|
+
/** @internal Aggiorna lo stato dei breadcrumb nel service dalla rotta corrente. */
|
|
49679
49676
|
refreshBreadcrumbs() {
|
|
49680
|
-
this.
|
|
49681
|
-
}
|
|
49682
|
-
/** @internal Determines the back navigation target from breadcrumbs. */
|
|
49683
|
-
getBackTarget() {
|
|
49684
|
-
if (this.breadcrumbs.length < 2)
|
|
49685
|
-
return null;
|
|
49686
|
-
return this.breadcrumbs[this.breadcrumbs.length - 2];
|
|
49677
|
+
this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
|
|
49687
49678
|
}
|
|
49688
|
-
/** @internal
|
|
49689
|
-
|
|
49690
|
-
const
|
|
49691
|
-
if (
|
|
49692
|
-
|
|
49679
|
+
/** @internal Determina se il back button deve essere nascosto per la rotta corrente. */
|
|
49680
|
+
shouldHideBackButton() {
|
|
49681
|
+
const currentPath = this.router.url.split('?')[0].split('#')[0];
|
|
49682
|
+
if (currentPath === this.homeRoute) {
|
|
49683
|
+
return true;
|
|
49693
49684
|
}
|
|
49694
|
-
|
|
49695
|
-
|
|
49685
|
+
let deepestRoute = this.activatedRoute;
|
|
49686
|
+
while (deepestRoute.firstChild) {
|
|
49687
|
+
deepestRoute = deepestRoute.firstChild;
|
|
49688
|
+
}
|
|
49689
|
+
// biome-ignore lint/complexity/useLiteralKeys: <explanation>
|
|
49690
|
+
if (deepestRoute.snapshot.data?.['hideBackButton'] === true) {
|
|
49691
|
+
return true;
|
|
49696
49692
|
}
|
|
49693
|
+
return false;
|
|
49694
|
+
}
|
|
49695
|
+
/** @internal Naviga al breadcrumb genitore o esegue il fallback alla home route. */
|
|
49696
|
+
navigateBack() {
|
|
49697
|
+
this.breadcrumbService.navigateBack(this.homeRoute);
|
|
49697
49698
|
}
|
|
49698
49699
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
49699
49700
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiBaseLayoutComponent, isStandalone: true, selector: "ui-base-layout", inputs: { showHeader: "showHeader", title: "title", homeRoute: "homeRoute", showHome: "showHome", updateDocumentTitle: "updateDocumentTitle", titleSuffix: "titleSuffix", showBackButton: "showBackButton", backButtonConfig: "backButtonConfig", actions: "actions", footerAlign: "footerAlign", footerGap: "footerGap" }, host: { classAttribute: "ui-base-layout-host" }, ngImport: i0, template: `
|