@mediusinc/mng-commons-layout 5.2.0-rc.2 → 5.3.0-rc.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.
@@ -1,9 +1,10 @@
1
- import { DestroyRef, Injectable, inject } from '@angular/core';
1
+ import { DestroyRef, Injectable, inject, signal } from '@angular/core';
2
2
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
3
  import { NavigationEnd, Router } from '@angular/router';
4
- import { ReplaySubject, Subject, take } from 'rxjs';
5
- import { filter } from 'rxjs/operators';
4
+ import { ReplaySubject, startWith, take } from 'rxjs';
5
+ import { filter, map } from 'rxjs/operators';
6
6
  import { CommonsRouterService, LoggerService, adjustRouteMenuLazyChildrenRouterLinks, doesUrlMatchRouterLink } from '@mediusinc/mng-commons/core';
7
+ import { prepareMenuItemsToInternal } from '../helpers/menu-items';
7
8
  import * as i0 from "@angular/core";
8
9
  export class MenuService {
9
10
  constructor() {
@@ -11,50 +12,45 @@ export class MenuService {
11
12
  this.logger = LoggerService.create('MenuService');
12
13
  this.router = inject(Router);
13
14
  this.commonsRouter = inject(CommonsRouterService);
14
- this.menuSource = new ReplaySubject(1);
15
- this.resetSource = new Subject();
16
- this.menuSource$ = this.menuSource.asObservable();
17
- this.resetSource$ = this.resetSource.asObservable();
18
- this.menuItems = [];
15
+ this._menuItems = signal([]);
16
+ this.menuItems = this._menuItems.asReadonly();
17
+ this.menuChangeSubject = new ReplaySubject(1);
18
+ this.menuChange$ = this.menuChangeSubject.asObservable();
19
19
  this.routeLoadedChildrenSubscriptions = [];
20
20
  }
21
- initialize(menuItems) {
22
- this.menuItems = menuItems;
21
+ setItems(menuItems) {
23
22
  this.routeLoadedChildrenSubscriptions.forEach(s => s.unsubscribe());
24
23
  this.routeLoadedChildrenSubscriptions = [];
25
- this.generateMenuItemKeys(menuItems);
24
+ this._menuItems.set(prepareMenuItemsToInternal(menuItems));
25
+ this.appendListenersToLazyChildren(this._menuItems());
26
+ this.router.events;
26
27
  this.router.events
27
- .pipe(filter(event => event instanceof NavigationEnd), takeUntilDestroyed(this.destroyRef))
28
+ .pipe(startWith(new NavigationEnd(0, this.router.url, this.router.url)), filter(event => event instanceof NavigationEnd), map(e => e), takeUntilDestroyed(this.destroyRef))
28
29
  .subscribe(e => {
29
30
  this.findAndSetActiveMenuItem(e.urlAfterRedirects);
30
31
  });
31
- this.findAndSetActiveMenuItem(this.router.url);
32
32
  }
33
- generateMenuItemKeys(menuItems, parentKey = '') {
34
- menuItems.forEach((item, idx) => {
35
- if (!item.index) {
36
- item.index = idx;
37
- item.key = parentKey ? parentKey + '-' + idx : idx.toString();
38
- }
39
- if (item.items) {
40
- this.generateMenuItemKeys(item.items, item.key);
41
- }
42
- else if (item.lazyChildren) {
33
+ appendListenersToLazyChildren(menuItems) {
34
+ menuItems.forEach(item => {
35
+ if (item.lazyChildren()) {
43
36
  this.listenToMenuItemLazyChildrenLoad(item);
44
37
  }
38
+ else if (item.items()) {
39
+ this.appendListenersToLazyChildren(item.items());
40
+ }
45
41
  });
46
42
  }
47
43
  findAndSetActiveMenuItem(url) {
48
- const matches = this.findActiveRouteMatches(this.menuItems, url);
44
+ const matches = this.findActiveRouteMatches(this.menuItems(), url);
49
45
  if (matches.length === 0) {
50
46
  this.reset();
51
47
  }
52
48
  else {
53
49
  const matchKey = matches[0][matches[0].length - 1].key;
54
50
  if (matchKey) {
55
- this.menuSource.next({
51
+ this.menuChangeSubject.next({
56
52
  key: matchKey,
57
- routeEvent: true
53
+ eventType: 'routeChange'
58
54
  });
59
55
  }
60
56
  else {
@@ -72,13 +68,14 @@ export class MenuService {
72
68
  .pipe(take(1))
73
69
  .subscribe({
74
70
  next: routes => {
75
- let menuItems = routes
71
+ const menuItemsFromRoutes = routes
76
72
  .filter(r => Array.isArray(r.data?.menuItems))
77
73
  .flatMap(r => r.data.menuItems);
78
- menuItems = adjustRouteMenuLazyChildrenRouterLinks(menuItem, menuItems);
79
- menuItem.items = menuItems;
80
- menuItem.lazyChildren = false;
81
- this.generateMenuItemKeys(menuItem.items, menuItem.key);
74
+ const menuItems = adjustRouteMenuLazyChildrenRouterLinks(menuItem, menuItemsFromRoutes);
75
+ menuItem.items.set(prepareMenuItemsToInternal(menuItems, menuItem.key));
76
+ this.appendListenersToLazyChildren(menuItem.items());
77
+ menuItem.itemsVisibility.set(menuItem.items()?.map(i => signal(true)));
78
+ menuItem.lazyChildren.set(false);
82
79
  this.findAndSetActiveMenuItem(this.router.url);
83
80
  }
84
81
  }));
@@ -86,11 +83,12 @@ export class MenuService {
86
83
  findActiveRouteMatches(menuItems, url) {
87
84
  const matches = [];
88
85
  for (const menuItem of menuItems) {
86
+ const chidldren = menuItem.items();
89
87
  if (menuItem.routerLink) {
90
88
  const isActive = doesUrlMatchRouterLink(menuItem.routerLink, url);
91
89
  if (isActive) {
92
- if (Array.isArray(menuItem.items)) {
93
- const itemsMatches = this.findActiveRouteMatches(menuItem.items, url);
90
+ if (Array.isArray(chidldren)) {
91
+ const itemsMatches = this.findActiveRouteMatches(chidldren, url);
94
92
  if (itemsMatches.length > 0) {
95
93
  matches.push(...itemsMatches.map(c => [menuItem, ...c]));
96
94
  }
@@ -103,8 +101,8 @@ export class MenuService {
103
101
  }
104
102
  }
105
103
  }
106
- else if (Array.isArray(menuItem.items)) {
107
- const itemsMatches = this.findActiveRouteMatches(menuItem.items, url);
104
+ else if (Array.isArray(chidldren)) {
105
+ const itemsMatches = this.findActiveRouteMatches(chidldren, url);
108
106
  if (itemsMatches.length > 0) {
109
107
  matches.push(...itemsMatches.map(c => [menuItem, ...c]));
110
108
  }
@@ -113,7 +111,9 @@ export class MenuService {
113
111
  return matches;
114
112
  }
115
113
  reset() {
116
- this.resetSource.next(true);
114
+ this.menuChangeSubject.next({
115
+ eventType: 'reset'
116
+ });
117
117
  }
118
118
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: MenuService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
119
119
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: MenuService }); }
@@ -121,4 +121,4 @@ export class MenuService {
121
121
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: MenuService, decorators: [{
122
122
  type: Injectable
123
123
  }] });
124
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu.service.js","sourceRoot":"","sources":["../../../../src/lib/services/menu.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DestroyRef, Injectable, inject} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {NavigationEnd, Router} from '@angular/router';\n\nimport {ReplaySubject, Subject, Subscription, take} from 'rxjs';\nimport {filter} from 'rxjs/operators';\n\nimport {CommonsMenuItem, CommonsRouteData, CommonsRouterService, LoggerService, adjustRouteMenuLazyChildrenRouterLinks, doesUrlMatchRouterLink} from '@mediusinc/mng-commons/core';\n\nimport {MenuChangeEvent} from '../models/menu.model';\n\n@Injectable()\nexport class MenuService {\n    private readonly destroyRef = inject(DestroyRef);\n    private readonly logger = LoggerService.create('MenuService');\n    private readonly router = inject(Router);\n    private readonly commonsRouter = inject(CommonsRouterService);\n    private readonly menuSource = new ReplaySubject<MenuChangeEvent>(1);\n    private readonly resetSource = new Subject();\n\n    menuSource$ = this.menuSource.asObservable();\n    resetSource$ = this.resetSource.asObservable();\n\n    private menuItems: Array<CommonsMenuItem> = [];\n    private routeLoadedChildrenSubscriptions: Subscription[] = [];\n\n    initialize(menuItems: Array<CommonsMenuItem>) {\n        this.menuItems = menuItems;\n\n        this.routeLoadedChildrenSubscriptions.forEach(s => s.unsubscribe());\n        this.routeLoadedChildrenSubscriptions = [];\n\n        this.generateMenuItemKeys(menuItems);\n\n        this.router.events\n            .pipe(\n                filter(event => event instanceof NavigationEnd),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(e => {\n                this.findAndSetActiveMenuItem((e as NavigationEnd).urlAfterRedirects);\n            });\n\n        this.findAndSetActiveMenuItem(this.router.url);\n    }\n\n    private generateMenuItemKeys(menuItems: Array<CommonsMenuItem>, parentKey = '') {\n        menuItems.forEach((item, idx) => {\n            if (!item.index) {\n                item.index = idx;\n                item.key = parentKey ? parentKey + '-' + idx : idx.toString();\n            }\n\n            if (item.items) {\n                this.generateMenuItemKeys(item.items, item.key);\n            } else if (item.lazyChildren) {\n                this.listenToMenuItemLazyChildrenLoad(item);\n            }\n        });\n    }\n\n    private findAndSetActiveMenuItem(url: string) {\n        const matches = this.findActiveRouteMatches(this.menuItems, url);\n        if (matches.length === 0) {\n            this.reset();\n        } else {\n            const matchKey = matches[0][matches[0].length - 1].key;\n            if (matchKey) {\n                this.menuSource.next({\n                    key: matchKey,\n                    routeEvent: true\n                });\n            } else {\n                this.logger.warn(`Found active menu item for url ${url}, but item key is not present`, matches);\n                this.reset();\n            }\n\n            if (matches.length > 1) {\n                this.logger.warn(`Multiple active menu items found for url ${url}`, matches);\n            }\n        }\n    }\n\n    private listenToMenuItemLazyChildrenLoad(menuItem: CommonsMenuItem) {\n        this.routeLoadedChildrenSubscriptions.push(\n            this.commonsRouter\n                .getRouteLoadedChildren$(menuItem.routerLink ?? '/')\n                .pipe(take(1))\n                .subscribe({\n                    next: routes => {\n                        let menuItems = routes\n                            .filter(r => Array.isArray((r.data as CommonsRouteData | undefined)?.menuItems))\n                            .flatMap(r => (r.data as CommonsRouteData).menuItems!);\n                        menuItems = adjustRouteMenuLazyChildrenRouterLinks(menuItem, menuItems);\n\n                        menuItem.items = menuItems;\n                        menuItem.lazyChildren = false;\n\n                        this.generateMenuItemKeys(menuItem.items, menuItem.key);\n                        this.findAndSetActiveMenuItem(this.router.url);\n                    }\n                })\n        );\n    }\n\n    private findActiveRouteMatches(menuItems: Array<CommonsMenuItem>, url: string): CommonsMenuItem[][] {\n        const matches: CommonsMenuItem[][] = [];\n        for (const menuItem of menuItems) {\n            if (menuItem.routerLink) {\n                const isActive = doesUrlMatchRouterLink(menuItem.routerLink, url);\n                if (isActive) {\n                    if (Array.isArray(menuItem.items)) {\n                        const itemsMatches = this.findActiveRouteMatches(menuItem.items, url);\n                        if (itemsMatches.length > 0) {\n                            matches.push(...itemsMatches.map(c => [menuItem, ...c]));\n                        } else {\n                            matches.push([menuItem]);\n                        }\n                    } else {\n                        matches.push([menuItem]);\n                    }\n                }\n            } else if (Array.isArray(menuItem.items)) {\n                const itemsMatches = this.findActiveRouteMatches(menuItem.items, url);\n                if (itemsMatches.length > 0) {\n                    matches.push(...itemsMatches.map(c => [menuItem, ...c]));\n                }\n            }\n        }\n        return matches;\n    }\n\n    reset() {\n        this.resetSource.next(true);\n    }\n}\n"]}
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu.service.js","sourceRoot":"","sources":["../../../../src/lib/services/menu.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DestroyRef, Injectable, inject, signal} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {NavigationEnd, Router} from '@angular/router';\n\nimport {ReplaySubject, Subscription, startWith, take} from 'rxjs';\nimport {filter, map} from 'rxjs/operators';\n\nimport {\n    CommonsMenuItem,\n    CommonsRouteData,\n    CommonsRouterService,\n    InternalCommonsMenuItem,\n    LoggerService,\n    adjustRouteMenuLazyChildrenRouterLinks,\n    doesUrlMatchRouterLink\n} from '@mediusinc/mng-commons/core';\n\nimport {prepareMenuItemsToInternal} from '../helpers/menu-items';\nimport {MenuChangeEvent} from '../models/menu.model';\n\n@Injectable()\nexport class MenuService {\n    private readonly destroyRef = inject(DestroyRef);\n    private readonly logger = LoggerService.create('MenuService');\n    private readonly router = inject(Router);\n    private readonly commonsRouter = inject(CommonsRouterService);\n\n    private _menuItems = signal<InternalCommonsMenuItem[]>([]);\n    public menuItems = this._menuItems.asReadonly();\n\n    private readonly menuChangeSubject = new ReplaySubject<MenuChangeEvent>(1);\n    menuChange$ = this.menuChangeSubject.asObservable();\n\n    private routeLoadedChildrenSubscriptions: Subscription[] = [];\n\n    setItems(menuItems: Array<CommonsMenuItem>) {\n        this.routeLoadedChildrenSubscriptions.forEach(s => s.unsubscribe());\n        this.routeLoadedChildrenSubscriptions = [];\n\n        this._menuItems.set(prepareMenuItemsToInternal(menuItems));\n        this.appendListenersToLazyChildren(this._menuItems());\n\n        this.router.events;\n        this.router.events\n            .pipe(\n                startWith(new NavigationEnd(0, this.router.url, this.router.url)),\n                filter(event => event instanceof NavigationEnd),\n                map(e => e as NavigationEnd),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(e => {\n                this.findAndSetActiveMenuItem((e as NavigationEnd).urlAfterRedirects);\n            });\n    }\n\n    private appendListenersToLazyChildren(menuItems: InternalCommonsMenuItem[]) {\n        menuItems.forEach(item => {\n            if (item.lazyChildren()) {\n                this.listenToMenuItemLazyChildrenLoad(item);\n            } else if (item.items()) {\n                this.appendListenersToLazyChildren(item.items());\n            }\n        });\n    }\n\n    private findAndSetActiveMenuItem(url: string) {\n        const matches = this.findActiveRouteMatches(this.menuItems(), url);\n        if (matches.length === 0) {\n            this.reset();\n        } else {\n            const matchKey = matches[0][matches[0].length - 1].key;\n            if (matchKey) {\n                this.menuChangeSubject.next({\n                    key: matchKey,\n                    eventType: 'routeChange'\n                });\n            } else {\n                this.logger.warn(`Found active menu item for url ${url}, but item key is not present`, matches);\n                this.reset();\n            }\n\n            if (matches.length > 1) {\n                this.logger.warn(`Multiple active menu items found for url ${url}`, matches);\n            }\n        }\n    }\n\n    private listenToMenuItemLazyChildrenLoad(menuItem: InternalCommonsMenuItem) {\n        this.routeLoadedChildrenSubscriptions.push(\n            this.commonsRouter\n                .getRouteLoadedChildren$(menuItem.routerLink ?? '/')\n                .pipe(take(1))\n                .subscribe({\n                    next: routes => {\n                        const menuItemsFromRoutes = routes\n                            .filter(r => Array.isArray((r.data as CommonsRouteData | undefined)?.menuItems))\n                            .flatMap(r => (r.data as CommonsRouteData).menuItems!);\n                        const menuItems = adjustRouteMenuLazyChildrenRouterLinks(menuItem, menuItemsFromRoutes);\n\n                        menuItem.items.set(prepareMenuItemsToInternal(menuItems, menuItem.key));\n                        this.appendListenersToLazyChildren(menuItem.items());\n                        menuItem.itemsVisibility.set(menuItem.items()?.map(i => signal(true)));\n                        menuItem.lazyChildren.set(false);\n\n                        this.findAndSetActiveMenuItem(this.router.url);\n                    }\n                })\n        );\n    }\n\n    private findActiveRouteMatches(menuItems: Array<InternalCommonsMenuItem>, url: string): InternalCommonsMenuItem[][] {\n        const matches: InternalCommonsMenuItem[][] = [];\n        for (const menuItem of menuItems) {\n            const chidldren = menuItem.items();\n            if (menuItem.routerLink) {\n                const isActive = doesUrlMatchRouterLink(menuItem.routerLink, url);\n                if (isActive) {\n                    if (Array.isArray(chidldren)) {\n                        const itemsMatches = this.findActiveRouteMatches(chidldren, url);\n                        if (itemsMatches.length > 0) {\n                            matches.push(...itemsMatches.map(c => [menuItem, ...c]));\n                        } else {\n                            matches.push([menuItem]);\n                        }\n                    } else {\n                        matches.push([menuItem]);\n                    }\n                }\n            } else if (Array.isArray(chidldren)) {\n                const itemsMatches = this.findActiveRouteMatches(chidldren, url);\n                if (itemsMatches.length > 0) {\n                    matches.push(...itemsMatches.map(c => [menuItem, ...c]));\n                }\n            }\n        }\n        return matches;\n    }\n\n    reset() {\n        this.menuChangeSubject.next({\n            eventType: 'reset'\n        });\n    }\n}\n"]}