@mediusinc/mng-commons-layout 5.2.0 → 5.3.0-rc.1
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/index.mjs +5 -2
- package/esm2022/lib/components/footer.component.mjs +5 -3
- package/esm2022/lib/components/main-layout.component.mjs +11 -13
- package/esm2022/lib/components/menu-item.component.mjs +64 -144
- package/esm2022/lib/components/menu.component.mjs +10 -13
- package/esm2022/lib/components/pages/error/error.page.component.mjs +20 -0
- package/esm2022/lib/components/pages/not-found/not-found.page.component.mjs +20 -0
- package/esm2022/lib/components/settings.component.mjs +89 -0
- package/esm2022/lib/components/sidebar.component.mjs +11 -12
- package/esm2022/lib/components/topbar-user.component.mjs +23 -4
- package/esm2022/lib/components/topbar.component.mjs +4 -4
- package/esm2022/lib/helpers/menu-items.mjs +26 -0
- package/esm2022/lib/models/menu.model.mjs +1 -1
- package/esm2022/lib/provide.mjs +3 -1
- package/esm2022/lib/services/layout-feature-config.token.mjs +1 -1
- package/esm2022/lib/services/layout.service.mjs +166 -0
- package/esm2022/lib/services/menu.service.mjs +37 -37
- package/fesm2022/mediusinc-mng-commons-layout.mjs +437 -287
- package/fesm2022/mediusinc-mng-commons-layout.mjs.map +1 -1
- package/index.d.ts +4 -1
- package/lib/components/footer.component.d.ts +2 -0
- package/lib/components/main-layout.component.d.ts +2 -5
- package/lib/components/menu-item.component.d.ts +15 -27
- package/lib/components/menu.component.d.ts +3 -4
- package/lib/components/pages/error/error.page.component.d.ts +7 -0
- package/lib/components/pages/not-found/not-found.page.component.d.ts +7 -0
- package/lib/components/settings.component.d.ts +30 -0
- package/lib/components/sidebar.component.d.ts +2 -4
- package/lib/components/topbar-user.component.d.ts +5 -0
- package/lib/components/topbar.component.d.ts +2 -2
- package/lib/helpers/menu-items.d.ts +2 -0
- package/lib/models/menu.model.d.ts +2 -2
- package/lib/services/layout-feature-config.token.d.ts +73 -0
- package/lib/services/layout.service.d.ts +46 -0
- package/lib/services/menu.service.d.ts +7 -8
- package/package.json +2 -2
- package/scss/layout/mng/_mng_layout_styles.scss +1 -0
- package/scss/layout/mng/_mng_sidebar_vertical.scss +8 -0
- package/version-info.json +6 -6
- package/esm2022/lib/services/main-layout.component.service.mjs +0 -71
- package/lib/services/main-layout.component.service.d.ts +0 -17
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
import { DOCUMENT } from '@angular/common';
|
|
2
|
+
import { Injectable, PLATFORM_ID, afterNextRender, computed, effect, inject, signal } from '@angular/core';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
import { COMMONS_MODULE_CONFIG_IT, CommonsStorageService } from '@mediusinc/mng-commons/core';
|
|
5
|
+
import { COMMONS_LAYOUT_FEATURE_CONFIG_IT } from './layout-feature-config.token';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
export class LayoutService {
|
|
8
|
+
getInitColorScheme() {
|
|
9
|
+
let colorScheme = this.storageService.getItem(this.typeName, this.storageColorSchemeKey);
|
|
10
|
+
if (colorScheme == null || !['dark', 'light', 'auto'].includes(colorScheme)) {
|
|
11
|
+
colorScheme = this.getDefaultColorScheme();
|
|
12
|
+
}
|
|
13
|
+
return colorScheme;
|
|
14
|
+
}
|
|
15
|
+
getInitActualColorScheme(colorScheme) {
|
|
16
|
+
if (colorScheme === 'auto') {
|
|
17
|
+
return window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';
|
|
18
|
+
}
|
|
19
|
+
else {
|
|
20
|
+
return colorScheme;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
getDefaultColorScheme() {
|
|
24
|
+
return this.config?.colorScheme ?? this.commonsConfig?.app?.colorScheme ?? 'auto';
|
|
25
|
+
}
|
|
26
|
+
setColorScheme(scheme) {
|
|
27
|
+
const defaultColorScheme = this.getDefaultColorScheme();
|
|
28
|
+
if (scheme === defaultColorScheme) {
|
|
29
|
+
this.storageService.removeItem(this.typeName, this.storageColorSchemeKey);
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
this.storageService.setItem(this.typeName, this.storageColorSchemeKey, scheme);
|
|
33
|
+
}
|
|
34
|
+
this._colorSchemeSetting.set(scheme);
|
|
35
|
+
this._colorScheme.set(this.getInitActualColorScheme(scheme));
|
|
36
|
+
}
|
|
37
|
+
constructor() {
|
|
38
|
+
this.typeName = 'MainLayoutComponentService';
|
|
39
|
+
this.storageMenuModeKey = 'menuMode';
|
|
40
|
+
this.storageColorSchemeKey = 'colorScheme';
|
|
41
|
+
this.platformId = inject(PLATFORM_ID);
|
|
42
|
+
this.document = inject(DOCUMENT);
|
|
43
|
+
this.commonsConfig = inject(COMMONS_MODULE_CONFIG_IT, { optional: true });
|
|
44
|
+
this.config = inject(COMMONS_LAYOUT_FEATURE_CONFIG_IT, { optional: true });
|
|
45
|
+
this.storageService = inject(CommonsStorageService);
|
|
46
|
+
this.state = signal({
|
|
47
|
+
staticMenuDesktopInactive: false,
|
|
48
|
+
overlayMenuActive: false,
|
|
49
|
+
rightMenuActive: false,
|
|
50
|
+
configSidebarVisible: false,
|
|
51
|
+
staticMenuMobileActive: false,
|
|
52
|
+
menuHoverActive: false,
|
|
53
|
+
sidebarActive: false,
|
|
54
|
+
anchored: false
|
|
55
|
+
});
|
|
56
|
+
this.overlayOpen = new Subject();
|
|
57
|
+
this.overlayOpen$ = this.overlayOpen.asObservable();
|
|
58
|
+
// color scheme
|
|
59
|
+
this._colorSchemeSetting = signal(this.getInitColorScheme());
|
|
60
|
+
this.colorSchemeSetting = this._colorSchemeSetting.asReadonly();
|
|
61
|
+
this._colorScheme = signal(this.getInitActualColorScheme(this.colorSchemeSetting()));
|
|
62
|
+
this.colorScheme = this._colorScheme.asReadonly();
|
|
63
|
+
this.colorSchemeIsLight = computed(() => this._colorScheme() === 'light');
|
|
64
|
+
// menu mode
|
|
65
|
+
this.menuModes = this.config?.menuModes ?? ['static', 'overlay', 'reveal', 'drawer', 'slim', 'slim-plus'];
|
|
66
|
+
this._menuMode = signal(this.initMenuMode());
|
|
67
|
+
this.menuMode = this._menuMode.asReadonly();
|
|
68
|
+
// logos
|
|
69
|
+
this.appLogoLight = signal(this.config?.logoPathLight ?? this.commonsConfig?.app?.logoPathLight ?? this.config?.logoPath ?? this.commonsConfig?.app?.logoPath ?? 'assets/layout/images/logo.png');
|
|
70
|
+
this.appLogoDark = signal(this.config?.logoPathLight ?? this.commonsConfig?.app?.logoPathDark ?? this.config?.logoPath ?? this.commonsConfig?.app?.logoPath ?? 'assets/layout/images/logo.png');
|
|
71
|
+
this.appLogo = computed(() => (this.colorSchemeIsLight() ? this.appLogoLight() : this.appLogoDark()));
|
|
72
|
+
this.appLogoNameLight = signal(this.config?.logoNamePathLight ??
|
|
73
|
+
this.commonsConfig?.app?.logoNamePathLight ??
|
|
74
|
+
this.config?.logoNamePath ??
|
|
75
|
+
this.commonsConfig?.app?.logoNamePath ??
|
|
76
|
+
'assets/layout/images/logo-appname.png');
|
|
77
|
+
this.appLogoNameDark = signal(this.config?.logoNamePathDark ??
|
|
78
|
+
this.commonsConfig?.app?.logoNamePathDark ??
|
|
79
|
+
this.config?.logoNamePath ??
|
|
80
|
+
this.commonsConfig?.app?.logoNamePath ??
|
|
81
|
+
'assets/layout/images/logo-appname.png');
|
|
82
|
+
this.appLogoName = computed(() => (this.colorSchemeIsLight() ? this.appLogoNameLight() : this.appLogoNameDark()));
|
|
83
|
+
// overlays
|
|
84
|
+
this.isOverlay = computed(() => this.menuMode() === 'overlay');
|
|
85
|
+
this.isSlim = computed(() => this.menuMode() === 'slim');
|
|
86
|
+
this.isSlimPlus = computed(() => this.menuMode() === 'slim-plus');
|
|
87
|
+
afterNextRender(() => {
|
|
88
|
+
window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', event => {
|
|
89
|
+
if (this.colorSchemeSetting() === 'auto') {
|
|
90
|
+
const newColorScheme = event.matches ? 'dark' : 'light';
|
|
91
|
+
this._colorScheme.set(newColorScheme);
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
effect(() => {
|
|
96
|
+
const colorScheme = this._colorScheme();
|
|
97
|
+
const linkElement = this.document.getElementById('app-theme');
|
|
98
|
+
if (colorScheme === 'dark' && linkElement.href.includes('light')) {
|
|
99
|
+
linkElement.href = 'theme-dark.css';
|
|
100
|
+
}
|
|
101
|
+
else if (colorScheme === 'light' && linkElement.href.includes('dark')) {
|
|
102
|
+
linkElement.href = 'theme-light.css';
|
|
103
|
+
}
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
onMenuModeChange(mode) {
|
|
107
|
+
if (!this.menuModes.includes(mode)) {
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
if (mode === this.config?.menuMode) {
|
|
111
|
+
this.storageService.removeItem(this.typeName, this.storageMenuModeKey);
|
|
112
|
+
}
|
|
113
|
+
else {
|
|
114
|
+
this.storageService.setItem(this.typeName, this.storageMenuModeKey, mode);
|
|
115
|
+
}
|
|
116
|
+
this._menuMode.set(mode);
|
|
117
|
+
}
|
|
118
|
+
onMenuToggle() {
|
|
119
|
+
if (this.isOverlay()) {
|
|
120
|
+
this.state.update(state => ({
|
|
121
|
+
...state,
|
|
122
|
+
overlayMenuActive: !state.overlayMenuActive
|
|
123
|
+
}));
|
|
124
|
+
if (this.state().overlayMenuActive) {
|
|
125
|
+
this.overlayOpen.next(null);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
if (this.isDesktop()) {
|
|
129
|
+
this.state.update(state => ({
|
|
130
|
+
...state,
|
|
131
|
+
staticMenuDesktopInactive: !state.staticMenuDesktopInactive
|
|
132
|
+
}));
|
|
133
|
+
}
|
|
134
|
+
else {
|
|
135
|
+
this.state.update(state => ({
|
|
136
|
+
...state,
|
|
137
|
+
staticMenuMobileActive: !state.staticMenuMobileActive
|
|
138
|
+
}));
|
|
139
|
+
if (this.state().staticMenuMobileActive) {
|
|
140
|
+
this.overlayOpen.next(null);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
onOverlaySubmenuOpen() {
|
|
145
|
+
this.overlayOpen.next(null);
|
|
146
|
+
}
|
|
147
|
+
isDesktop() {
|
|
148
|
+
return window.innerWidth > 991;
|
|
149
|
+
}
|
|
150
|
+
isMobile() {
|
|
151
|
+
return !this.isDesktop();
|
|
152
|
+
}
|
|
153
|
+
initMenuMode() {
|
|
154
|
+
const lsMode = this.storageService.getItem(this.typeName, this.storageMenuModeKey);
|
|
155
|
+
if (lsMode && this.menuModes.includes(lsMode)) {
|
|
156
|
+
return lsMode;
|
|
157
|
+
}
|
|
158
|
+
return this.config?.menuMode ?? 'static';
|
|
159
|
+
}
|
|
160
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: LayoutService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
161
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: LayoutService }); }
|
|
162
|
+
}
|
|
163
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: LayoutService, decorators: [{
|
|
164
|
+
type: Injectable
|
|
165
|
+
}], ctorParameters: () => [] });
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout.service.js","sourceRoot":"","sources":["../../../../src/lib/services/layout.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,QAAQ,EAAC,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAC,UAAU,EAAE,WAAW,EAAE,eAAe,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAC,MAAM,eAAe,CAAC;AAEzG,OAAO,EAAC,OAAO,EAAC,MAAM,MAAM,CAAC;AAE7B,OAAO,EAAC,wBAAwB,EAAE,qBAAqB,EAAC,MAAM,6BAA6B,CAAC;AAG5F,OAAO,EAAC,gCAAgC,EAA8B,MAAM,+BAA+B,CAAC;;AAG5G,MAAM,OAAO,aAAa;IAgCd,kBAAkB;QACtB,IAAI,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAc,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC;QACtG,IAAI,WAAW,IAAI,IAAI,IAAI,CAAE,CAAC,MAAM,EAAE,OAAO,EAAE,MAAM,CAAmB,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;YAC7F,WAAW,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/C,CAAC;QACD,OAAO,WAAW,CAAC;IACvB,CAAC;IAEO,wBAAwB,CAAC,WAAwB;QACrD,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YACzB,OAAO,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,UAAU,CAAC,8BAA8B,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC;QAC7G,CAAC;aAAM,CAAC;YACJ,OAAO,WAAW,CAAC;QACvB,CAAC;IACL,CAAC;IAEO,qBAAqB;QACzB,OAAO,IAAI,CAAC,MAAM,EAAE,WAAW,IAAI,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,WAAW,IAAI,MAAM,CAAC;IACtF,CAAC;IAEM,cAAc,CAAC,MAAmB;QACrC,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QACxD,IAAI,MAAM,KAAK,kBAAkB,EAAE,CAAC;YAChC,IAAI,CAAC,cAAc,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC;QAC9E,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,qBAAqB,EAAE,MAAM,CAAC,CAAC;QACnF,CAAC;QACD,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACrC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC,CAAC;IACjE,CAAC;IAqCD;QAjGiB,aAAQ,GAAG,4BAA4B,CAAC;QACxC,uBAAkB,GAAG,UAAU,CAAC;QAChC,0BAAqB,GAAG,aAAa,CAAC;QAEtC,eAAU,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QACjC,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC5B,kBAAa,GAAG,MAAM,CAAC,wBAAwB,EAAE,EAAC,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;QACnE,WAAM,GAAG,MAAM,CAAC,gCAAgC,EAAE,EAAC,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;QACpE,mBAAc,GAAG,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAEzD,UAAK,GAAG,MAAM,CAAc;YAC/B,yBAAyB,EAAE,KAAK;YAChC,iBAAiB,EAAE,KAAK;YACxB,eAAe,EAAE,KAAK;YACtB,oBAAoB,EAAE,KAAK;YAC3B,sBAAsB,EAAE,KAAK;YAC7B,eAAe,EAAE,KAAK;YACtB,aAAa,EAAE,KAAK;YACpB,QAAQ,EAAE,KAAK;SAClB,CAAC,CAAC;QAEK,gBAAW,GAAG,IAAI,OAAO,EAAO,CAAC;QAClC,iBAAY,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,EAAE,CAAC;QAEtD,eAAe;QACP,wBAAmB,GAAG,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC;QACzD,uBAAkB,GAAG,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAC;QAC1D,iBAAY,GAAG,MAAM,CAAiC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC;QACjH,gBAAW,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;QAC7C,uBAAkB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,CAAC,CAAC;QAiC5E,YAAY;QACI,cAAS,GAAqB,IAAI,CAAC,MAAM,EAAE,SAAS,IAAI,CAAC,QAAQ,EAAE,SAAS,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,WAAW,CAAC,CAAC;QAC/H,cAAS,GAAG,MAAM,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;QACzC,aAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,UAAU,EAAE,CAAC;QAE9C,QAAQ;QACA,iBAAY,GAAG,MAAM,CACzB,IAAI,CAAC,MAAM,EAAE,aAAa,IAAI,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,aAAa,IAAI,IAAI,CAAC,MAAM,EAAE,QAAQ,IAAI,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,QAAQ,IAAI,+BAA+B,CACxK,CAAC;QACM,gBAAW,GAAG,MAAM,CACxB,IAAI,CAAC,MAAM,EAAE,aAAa,IAAI,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,YAAY,IAAI,IAAI,CAAC,MAAM,EAAE,QAAQ,IAAI,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,QAAQ,IAAI,+BAA+B,CACvK,CAAC;QACK,YAAO,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QAEhG,qBAAgB,GAAG,MAAM,CAC7B,IAAI,CAAC,MAAM,EAAE,iBAAiB;YAC1B,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,iBAAiB;YAC1C,IAAI,CAAC,MAAM,EAAE,YAAY;YACzB,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,YAAY;YACrC,uCAAuC,CAC9C,CAAC;QACM,oBAAe,GAAG,MAAM,CAC5B,IAAI,CAAC,MAAM,EAAE,gBAAgB;YACzB,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,gBAAgB;YACzC,IAAI,CAAC,MAAM,EAAE,YAAY;YACzB,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,YAAY;YACrC,uCAAuC,CAC9C,CAAC;QACK,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QAEpH,WAAW;QACJ,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,SAAS,CAAC,CAAC;QAC1D,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,MAAM,CAAC,CAAC;QACpD,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,WAAW,CAAC,CAAC;QAGhE,eAAe,CAAC,GAAG,EAAE;YACjB,MAAM,CAAC,UAAU,CAAC,8BAA8B,CAAC,CAAC,gBAAgB,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAE;gBACjF,IAAI,IAAI,CAAC,kBAAkB,EAAE,KAAK,MAAM,EAAE,CAAC;oBACvC,MAAM,cAAc,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC;oBACxD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;gBAC1C,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,GAAG,EAAE;YACR,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACxC,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,WAAW,CAAoB,CAAC;YACjF,IAAI,WAAW,KAAK,MAAM,IAAI,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC/D,WAAW,CAAC,IAAI,GAAG,gBAAgB,CAAC;YACxC,CAAC;iBAAM,IAAI,WAAW,KAAK,OAAO,IAAI,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtE,WAAW,CAAC,IAAI,GAAG,iBAAiB,CAAC;YACzC,CAAC;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAED,gBAAgB,CAAC,IAAoB;QACjC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YACjC,OAAO;QACX,CAAC;QACD,IAAI,IAAI,KAAK,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAE,CAAC;YACjC,IAAI,CAAC,cAAc,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAC3E,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC;QAC9E,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC7B,CAAC;IAED,YAAY;QACR,IAAI,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;YACnB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBACxB,GAAG,KAAK;gBACR,iBAAiB,EAAE,CAAC,KAAK,CAAC,iBAAiB;aAC9C,CAAC,CAAC,CAAC;YAEJ,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,iBAAiB,EAAE,CAAC;gBACjC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,CAAC;QACL,CAAC;QAED,IAAI,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;YACnB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBACxB,GAAG,KAAK;gBACR,yBAAyB,EAAE,CAAC,KAAK,CAAC,yBAAyB;aAC9D,CAAC,CAAC,CAAC;QACR,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBACxB,GAAG,KAAK;gBACR,sBAAsB,EAAE,CAAC,KAAK,CAAC,sBAAsB;aACxD,CAAC,CAAC,CAAC;YAEJ,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,sBAAsB,EAAE,CAAC;gBACtC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,CAAC;QACL,CAAC;IACL,CAAC;IAED,oBAAoB;QAChB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAED,SAAS;QACL,OAAO,MAAM,CAAC,UAAU,GAAG,GAAG,CAAC;IACnC,CAAC;IAED,QAAQ;QACJ,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;IAC7B,CAAC;IAEO,YAAY;QAChB,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAiB,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;QACnG,IAAI,MAAM,IAAI,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YAC5C,OAAO,MAAM,CAAC;QAClB,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,EAAE,QAAQ,IAAI,QAAQ,CAAC;IAC7C,CAAC;8GAlLQ,aAAa;kHAAb,aAAa;;2FAAb,aAAa;kBADzB,UAAU","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {Injectable, PLATFORM_ID, afterNextRender, computed, effect, inject, signal} from '@angular/core';\n\nimport {Subject} from 'rxjs';\n\nimport {COMMONS_MODULE_CONFIG_IT, CommonsStorageService} from '@mediusinc/mng-commons/core';\n\nimport {LayoutState} from '../models/layout-state.model';\nimport {COMMONS_LAYOUT_FEATURE_CONFIG_IT, ColorScheme, LayoutMenuMode} from './layout-feature-config.token';\n\n@Injectable()\nexport class LayoutService {\n    private readonly typeName = 'MainLayoutComponentService';\n    private readonly storageMenuModeKey = 'menuMode';\n    private readonly storageColorSchemeKey = 'colorScheme';\n\n    private readonly platformId = inject(PLATFORM_ID);\n    private readonly document = inject(DOCUMENT);\n    private readonly commonsConfig = inject(COMMONS_MODULE_CONFIG_IT, {optional: true});\n    private readonly config = inject(COMMONS_LAYOUT_FEATURE_CONFIG_IT, {optional: true});\n    private readonly storageService = inject(CommonsStorageService);\n\n    public state = signal<LayoutState>({\n        staticMenuDesktopInactive: false,\n        overlayMenuActive: false,\n        rightMenuActive: false,\n        configSidebarVisible: false,\n        staticMenuMobileActive: false,\n        menuHoverActive: false,\n        sidebarActive: false,\n        anchored: false\n    });\n\n    private overlayOpen = new Subject<any>();\n    public overlayOpen$ = this.overlayOpen.asObservable();\n\n    // color scheme\n    private _colorSchemeSetting = signal(this.getInitColorScheme());\n    public colorSchemeSetting = this._colorSchemeSetting.asReadonly();\n    private _colorScheme = signal<Exclude<ColorScheme, 'system'>>(this.getInitActualColorScheme(this.colorSchemeSetting()));\n    public colorScheme = this._colorScheme.asReadonly();\n    public colorSchemeIsLight = computed(() => this._colorScheme() === 'light');\n\n    private getInitColorScheme(): ColorScheme {\n        let colorScheme = this.storageService.getItem<ColorScheme>(this.typeName, this.storageColorSchemeKey);\n        if (colorScheme == null || !(['dark', 'light', 'auto'] as ColorScheme[]).includes(colorScheme)) {\n            colorScheme = this.getDefaultColorScheme();\n        }\n        return colorScheme;\n    }\n\n    private getInitActualColorScheme(colorScheme: ColorScheme): Exclude<ColorScheme, 'system'> {\n        if (colorScheme === 'auto') {\n            return window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';\n        } else {\n            return colorScheme;\n        }\n    }\n\n    private getDefaultColorScheme(): ColorScheme {\n        return this.config?.colorScheme ?? this.commonsConfig?.app?.colorScheme ?? 'auto';\n    }\n\n    public setColorScheme(scheme: ColorScheme) {\n        const defaultColorScheme = this.getDefaultColorScheme();\n        if (scheme === defaultColorScheme) {\n            this.storageService.removeItem(this.typeName, this.storageColorSchemeKey);\n        } else {\n            this.storageService.setItem(this.typeName, this.storageColorSchemeKey, scheme);\n        }\n        this._colorSchemeSetting.set(scheme);\n        this._colorScheme.set(this.getInitActualColorScheme(scheme));\n    }\n\n    // menu mode\n    public readonly menuModes: LayoutMenuMode[] = this.config?.menuModes ?? ['static', 'overlay', 'reveal', 'drawer', 'slim', 'slim-plus'];\n    private _menuMode = signal(this.initMenuMode());\n    public menuMode = this._menuMode.asReadonly();\n\n    // logos\n    private appLogoLight = signal(\n        this.config?.logoPathLight ?? this.commonsConfig?.app?.logoPathLight ?? this.config?.logoPath ?? this.commonsConfig?.app?.logoPath ?? 'assets/layout/images/logo.png'\n    );\n    private appLogoDark = signal(\n        this.config?.logoPathLight ?? this.commonsConfig?.app?.logoPathDark ?? this.config?.logoPath ?? this.commonsConfig?.app?.logoPath ?? 'assets/layout/images/logo.png'\n    );\n    public appLogo = computed(() => (this.colorSchemeIsLight() ? this.appLogoLight() : this.appLogoDark()));\n\n    private appLogoNameLight = signal(\n        this.config?.logoNamePathLight ??\n            this.commonsConfig?.app?.logoNamePathLight ??\n            this.config?.logoNamePath ??\n            this.commonsConfig?.app?.logoNamePath ??\n            'assets/layout/images/logo-appname.png'\n    );\n    private appLogoNameDark = signal(\n        this.config?.logoNamePathDark ??\n            this.commonsConfig?.app?.logoNamePathDark ??\n            this.config?.logoNamePath ??\n            this.commonsConfig?.app?.logoNamePath ??\n            'assets/layout/images/logo-appname.png'\n    );\n    public appLogoName = computed(() => (this.colorSchemeIsLight() ? this.appLogoNameLight() : this.appLogoNameDark()));\n\n    // overlays\n    public isOverlay = computed(() => this.menuMode() === 'overlay');\n    public isSlim = computed(() => this.menuMode() === 'slim');\n    public isSlimPlus = computed(() => this.menuMode() === 'slim-plus');\n\n    constructor() {\n        afterNextRender(() => {\n            window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', event => {\n                if (this.colorSchemeSetting() === 'auto') {\n                    const newColorScheme = event.matches ? 'dark' : 'light';\n                    this._colorScheme.set(newColorScheme);\n                }\n            });\n        });\n\n        effect(() => {\n            const colorScheme = this._colorScheme();\n            const linkElement = this.document.getElementById('app-theme') as HTMLLinkElement;\n            if (colorScheme === 'dark' && linkElement.href.includes('light')) {\n                linkElement.href = 'theme-dark.css';\n            } else if (colorScheme === 'light' && linkElement.href.includes('dark')) {\n                linkElement.href = 'theme-light.css';\n            }\n        });\n    }\n\n    onMenuModeChange(mode: LayoutMenuMode) {\n        if (!this.menuModes.includes(mode)) {\n            return;\n        }\n        if (mode === this.config?.menuMode) {\n            this.storageService.removeItem(this.typeName, this.storageMenuModeKey);\n        } else {\n            this.storageService.setItem(this.typeName, this.storageMenuModeKey, mode);\n        }\n        this._menuMode.set(mode);\n    }\n\n    onMenuToggle() {\n        if (this.isOverlay()) {\n            this.state.update(state => ({\n                ...state,\n                overlayMenuActive: !state.overlayMenuActive\n            }));\n\n            if (this.state().overlayMenuActive) {\n                this.overlayOpen.next(null);\n            }\n        }\n\n        if (this.isDesktop()) {\n            this.state.update(state => ({\n                ...state,\n                staticMenuDesktopInactive: !state.staticMenuDesktopInactive\n            }));\n        } else {\n            this.state.update(state => ({\n                ...state,\n                staticMenuMobileActive: !state.staticMenuMobileActive\n            }));\n\n            if (this.state().staticMenuMobileActive) {\n                this.overlayOpen.next(null);\n            }\n        }\n    }\n\n    onOverlaySubmenuOpen() {\n        this.overlayOpen.next(null);\n    }\n\n    isDesktop() {\n        return window.innerWidth > 991;\n    }\n\n    isMobile() {\n        return !this.isDesktop();\n    }\n\n    private initMenuMode(): LayoutMenuMode {\n        const lsMode = this.storageService.getItem<LayoutMenuMode>(this.typeName, this.storageMenuModeKey);\n        if (lsMode && this.menuModes.includes(lsMode)) {\n            return lsMode;\n        }\n        return this.config?.menuMode ?? 'static';\n    }\n}\n"]}
|
|
@@ -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,
|
|
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.
|
|
15
|
-
this.
|
|
16
|
-
this.
|
|
17
|
-
this.
|
|
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
|
-
|
|
22
|
-
this.menuItems = menuItems;
|
|
21
|
+
setItems(menuItems) {
|
|
23
22
|
this.routeLoadedChildrenSubscriptions.forEach(s => s.unsubscribe());
|
|
24
23
|
this.routeLoadedChildrenSubscriptions = [];
|
|
25
|
-
this.
|
|
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
|
-
|
|
34
|
-
menuItems.forEach(
|
|
35
|
-
if (
|
|
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.
|
|
51
|
+
this.menuChangeSubject.next({
|
|
56
52
|
key: matchKey,
|
|
57
|
-
|
|
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
|
-
|
|
71
|
+
const menuItemsFromRoutes = routes
|
|
76
72
|
.filter(r => Array.isArray(r.data?.menuItems))
|
|
77
73
|
.flatMap(r => r.data.menuItems);
|
|
78
|
-
menuItems = adjustRouteMenuLazyChildrenRouterLinks(menuItem,
|
|
79
|
-
menuItem.items
|
|
80
|
-
menuItem.
|
|
81
|
-
|
|
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(
|
|
93
|
-
const itemsMatches = this.findActiveRouteMatches(
|
|
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(
|
|
107
|
-
const itemsMatches = this.findActiveRouteMatches(
|
|
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.
|
|
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"]}
|