@yuuvis/client-shell 0.6.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/README.md +94 -0
  2. package/esm2022/index.mjs +6 -0
  3. package/esm2022/lib/actions/manage-flavors/manage-flavors.action.mjs +39 -0
  4. package/esm2022/lib/actions/manage-flavors/manage-flavors.component.mjs +69 -0
  5. package/esm2022/lib/client-shell.component.mjs +191 -0
  6. package/esm2022/lib/components/app-logo/app-logo.component.mjs +18 -0
  7. package/esm2022/lib/directives/inert.directive.mjs +26 -0
  8. package/esm2022/lib/lib.routes.mjs +15 -0
  9. package/esm2022/lib/pages/dashboard/dashboard.component.mjs +11 -0
  10. package/esm2022/lib/pages/notifications/notifications.component.mjs +82 -0
  11. package/esm2022/lib/pages/settings/settings.component.mjs +71 -0
  12. package/esm2022/lib/pages/web-share-target/web-share-target.component.mjs +18 -0
  13. package/esm2022/widget-dashboard/index.mjs +3 -0
  14. package/esm2022/widget-dashboard/lib/widget-dashboard.component.mjs +46 -0
  15. package/esm2022/widget-dashboard/lib/widget-dashboard.config.mjs +3 -0
  16. package/esm2022/widget-dashboard/lib/widget-dashboard.module.mjs +49 -0
  17. package/esm2022/widget-dashboard/yuuvis-client-shell-widget-dashboard.mjs +5 -0
  18. package/esm2022/yuuvis-client-shell.mjs +5 -0
  19. package/fesm2022/yuuvis-client-shell-widget-dashboard.mjs +98 -0
  20. package/fesm2022/yuuvis-client-shell-widget-dashboard.mjs.map +1 -0
  21. package/fesm2022/yuuvis-client-shell.mjs +496 -0
  22. package/fesm2022/yuuvis-client-shell.mjs.map +1 -0
  23. package/index.d.ts +5 -0
  24. package/lib/actions/manage-flavors/manage-flavors.action.d.ts +17 -0
  25. package/lib/actions/manage-flavors/manage-flavors.component.d.ts +17 -0
  26. package/lib/assets/i18n/de.json +30 -0
  27. package/lib/assets/i18n/en.json +30 -0
  28. package/lib/client-shell.component.d.ts +42 -0
  29. package/lib/components/app-logo/app-logo.component.d.ts +7 -0
  30. package/lib/directives/inert.directive.d.ts +8 -0
  31. package/lib/lib.routes.d.ts +2 -0
  32. package/lib/pages/dashboard/dashboard.component.d.ts +5 -0
  33. package/lib/pages/notifications/notifications.component.d.ts +29 -0
  34. package/lib/pages/settings/settings.component.d.ts +28 -0
  35. package/lib/pages/web-share-target/web-share-target.component.d.ts +6 -0
  36. package/package.json +40 -0
  37. package/widget-dashboard/README.md +19 -0
  38. package/widget-dashboard/index.d.ts +2 -0
  39. package/widget-dashboard/lib/widget-dashboard.component.d.ts +16 -0
  40. package/widget-dashboard/lib/widget-dashboard.config.d.ts +7 -0
  41. package/widget-dashboard/lib/widget-dashboard.module.d.ts +13 -0
package/README.md ADDED
@@ -0,0 +1,94 @@
1
+ # @yuuvis/client-shell
2
+
3
+ The client shell project provides the frame for a yuuvis® MOMENTUM client application.
4
+ This frame is then able to host multiple dedicated apps built for the shell environment.
5
+ For the hosted apps `@yuuvis/client-shell` provides a set of services and components to interact with with a yuuvis® MOMENTUM backend.
6
+
7
+ ## User settings
8
+
9
+ Settings are handled by the shell. It provides a page where users can configure their application(s). As authentication is covered by the shell, settings will be managed as well. In general this includes e.g. the users preferred language, but also settings of each individual in-shell-app.
10
+
11
+ ## Interfaces
12
+
13
+ Client shell provides a couple of APIs apps can utilize.
14
+
15
+ ### Actions
16
+
17
+ Using the `actions.service` apps are able to register actions to the whole shell and its applications. Let's say we have an app dedicated to deal with eMails you can create an actions that will open any email (regardless of the app showing it) in your eMail app. To do so you first of all write the action (checkout `open-in-app.action.ts` for the code).
18
+
19
+ In your apps extension package you can now register that action:
20
+
21
+ ```ts
22
+ @Injectable()
23
+ export class ExtensionsService {
24
+ init(): Promise<any> {
25
+ this.#actionsService.register([{ id: ACTION_ID, action: MailArchiveOpenInAppAction }]);
26
+ }
27
+ }
28
+ ```
29
+
30
+ Now this action is made available to all apps.
31
+
32
+ ### Renderers
33
+
34
+ The objects apps are dealing with will have a variety of property types. The basic ones are defined by the schema definition. Here we have:
35
+
36
+ - string
37
+ - date
38
+ - integer
39
+ - decimal
40
+ - ...
41
+
42
+ Renderers are used to define how the values for those properties are rendered. Dates for example are formatted based on the users locale.
43
+
44
+ There may be special properties that need a different treatment. The `system:contentStreamLength` property of objects define there size in bytes. This would be hard to read for users so we can create a renderer for this particular property (checkout `filesize.renderer.component.ts` for the code).
45
+
46
+ In your apps extension package you can now register that renderer:
47
+
48
+ ```ts
49
+ @Injectable()
50
+ export class ExtensionsService {
51
+ init(): Promise<any> {
52
+ this.#rendererService.registerRenderer(FileSizeRendererComponent, `system:contentStreamLength`);
53
+ }
54
+ }
55
+ ```
56
+
57
+ Now this renderer is made available to all apps. You can limit the renderer to specific types by adding a third parameter to the registerRenderer function.
58
+
59
+ ### Object types
60
+
61
+ Types/Flavors are collections of properties that can be added to an object. Let's say you have a document that is an invoice. First of all any file is just a document. By adding a type of `invoice` its metadata will be extended by the properties defined by the aspect.
62
+
63
+ ```ts
64
+ @Injectable()
65
+ export class ExtensionsService {
66
+ #shell = inject(ShellService);
67
+
68
+ init(): Promise<any> {
69
+ this.#shell.exposeObjectFlavors(APP_FLAVORS);
70
+ }
71
+ }
72
+ ```
73
+
74
+ Flavors are simple objects defining a type:
75
+
76
+ ```ts
77
+ const flavor = {
78
+ // unique id
79
+ id: 'io.yuuvis.app.ai.flavor.invoice',
80
+ // key to fetch translated description
81
+ descriptionKey: 'io.yuuvis.app.ai.flavor.invoice.description',
82
+ // an icon
83
+ icon: '<svg/>...</svg>',
84
+ // the base object type ID for the type
85
+ objectTypeID: SystemType.DOCUMENT,
86
+ // ID of the secondary object type defining the properties of the type
87
+ sot: `${APP_PREFIX}invoice`,
88
+ // restrict on which kind of files this type could be applied
89
+ applicableTo: {
90
+ // match against mime type incl. wildcards
91
+ mimeTypes: ['*/*']
92
+ }
93
+ };
94
+ ```
@@ -0,0 +1,6 @@
1
+ export * from './lib/directives/inert.directive';
2
+ export * from './lib/client-shell.component';
3
+ export * from './lib/components/app-logo/app-logo.component';
4
+ export * from './lib/lib.routes';
5
+ export * from './lib/actions/manage-flavors/manage-flavors.action';
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3l1dXZpcy9jbGllbnQtc2hlbGwvc3JjL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsa0NBQWtDLENBQUM7QUFDakQsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLDhDQUE4QyxDQUFDO0FBQzdELGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxvREFBb0QsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbGliL2RpcmVjdGl2ZXMvaW5lcnQuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NsaWVudC1zaGVsbC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY29tcG9uZW50cy9hcHAtbG9nby9hcHAtbG9nby5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbGliLnJvdXRlcyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9hY3Rpb25zL21hbmFnZS1mbGF2b3JzL21hbmFnZS1mbGF2b3JzLmFjdGlvbic7XG5cbiJdfQ==
@@ -0,0 +1,39 @@
1
+ import { inject } from '@angular/core';
2
+ import { TranslateService } from '@yuuvis/client-core';
3
+ import { AbstractContextAction, ACTION_ICON, SelectionRange } from '@yuuvis/client-framework/actions';
4
+ import { ShellService } from '@yuuvis/client-shell-core';
5
+ import { YvcOverlayService } from '@yuuvis/components/overlay';
6
+ import { of } from 'rxjs';
7
+ import { ManageFlavorsComponent } from './manage-flavors.component';
8
+ export class ManageFlavorsAction extends AbstractContextAction {
9
+ constructor() {
10
+ super(...arguments);
11
+ this.#shell = inject(ShellService);
12
+ this.#overlay = inject(YvcOverlayService);
13
+ this.translate = inject(TranslateService);
14
+ this.id = 'yuv.base.manage-flavor';
15
+ this.label = this.translate.instant('yuv.shell.action.manage-flavor.label');
16
+ this.description = this.translate.instant('yuv.shell.action.manage-flavor.description');
17
+ this.priority = 8;
18
+ this.icon = ACTION_ICON.manageFlavor;
19
+ this.group = 'common';
20
+ this.range = SelectionRange.SINGLE_SELECT;
21
+ this.supports = {
22
+ pattern: '*'
23
+ };
24
+ }
25
+ #shell;
26
+ #overlay;
27
+ isExecutable(items) {
28
+ const item = items[0];
29
+ return of(item && !!item.permissions?.writeIndexData && !!item.content && this.#shell.getApplicableObjectFlavors(item.content.mimeType).length > 0);
30
+ }
31
+ run(items) {
32
+ this.#overlay.open(ManageFlavorsComponent, items[0], {
33
+ width: '400px',
34
+ maxWidth: '90vw',
35
+ });
36
+ return of(true);
37
+ }
38
+ }
39
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFuYWdlLWZsYXZvcnMuYWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy95dXV2aXMvY2xpZW50LXNoZWxsL3NyYy9saWIvYWN0aW9ucy9tYW5hZ2UtZmxhdm9ycy9tYW5hZ2UtZmxhdm9ycy5hY3Rpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN2QyxPQUFPLEVBQWEsZ0JBQWdCLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUNsRSxPQUFPLEVBQUUscUJBQXFCLEVBQVUsV0FBVyxFQUE4QixjQUFjLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUMxSSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDekQsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDL0QsT0FBTyxFQUFjLEVBQUUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUN0QyxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUVwRSxNQUFNLE9BQU8sbUJBQW9CLFNBQVEscUJBQXFCO0lBQTlEOztRQUNFLFdBQU0sR0FBRyxNQUFNLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDOUIsYUFBUSxHQUFHLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQzdCLGNBQVMsR0FBRyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUU3QyxPQUFFLEdBQUcsd0JBQXdCLENBQUM7UUFDOUIsVUFBSyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDLHNDQUFzQyxDQUFDLENBQUM7UUFDdkUsZ0JBQVcsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQyw0Q0FBNEMsQ0FBQyxDQUFDO1FBQ25GLGFBQVEsR0FBRyxDQUFDLENBQUM7UUFDYixTQUFJLEdBQUcsV0FBVyxDQUFDLFlBQVksQ0FBQztRQUNoQyxVQUFLLEdBQUcsUUFBUSxDQUFDO1FBQ2pCLFVBQUssR0FBRyxjQUFjLENBQUMsYUFBYSxDQUFDO1FBQ3JDLGFBQVEsR0FBa0I7WUFDeEIsT0FBTyxFQUFFLEdBQUc7U0FDYixDQUFDO0lBY0osQ0FBQztJQTNCQyxNQUFNLENBQXdCO0lBQzlCLFFBQVEsQ0FBNkI7SUFjckMsWUFBWSxDQUFDLEtBQWtCO1FBQzdCLE1BQU0sSUFBSSxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QixPQUFPLEVBQUUsQ0FBQyxJQUFJLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsY0FBYyxJQUFJLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsMEJBQTBCLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUM7SUFDdEosQ0FBQztJQUVELEdBQUcsQ0FBQyxLQUFrQjtRQUNwQixJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxzQkFBc0IsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUU7WUFDbkQsS0FBSyxFQUFFLE9BQU87WUFDZCxRQUFRLEVBQUUsTUFBTTtTQUNqQixDQUFDLENBQUM7UUFDSCxPQUFPLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNsQixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IERtc09iamVjdCwgVHJhbnNsYXRlU2VydmljZSB9IGZyb20gJ0B5dXV2aXMvY2xpZW50LWNvcmUnO1xuaW1wb3J0IHsgQWJzdHJhY3RDb250ZXh0QWN0aW9uLCBBY3Rpb24sIEFDVElPTl9JQ09OLCBBY3Rpb25TdXBwb3J0LCBCQVNFX0FDVElPTiwgU2VsZWN0aW9uUmFuZ2UgfSBmcm9tICdAeXV1dmlzL2NsaWVudC1mcmFtZXdvcmsvYWN0aW9ucyc7XG5pbXBvcnQgeyBTaGVsbFNlcnZpY2UgfSBmcm9tICdAeXV1dmlzL2NsaWVudC1zaGVsbC1jb3JlJztcbmltcG9ydCB7IFl2Y092ZXJsYXlTZXJ2aWNlIH0gZnJvbSAnQHl1dXZpcy9jb21wb25lbnRzL292ZXJsYXknO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgb2YgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IE1hbmFnZUZsYXZvcnNDb21wb25lbnQgfSBmcm9tICcuL21hbmFnZS1mbGF2b3JzLmNvbXBvbmVudCc7XG5cbmV4cG9ydCBjbGFzcyBNYW5hZ2VGbGF2b3JzQWN0aW9uIGV4dGVuZHMgQWJzdHJhY3RDb250ZXh0QWN0aW9uIGltcGxlbWVudHMgQWN0aW9uIHtcbiAgI3NoZWxsID0gaW5qZWN0KFNoZWxsU2VydmljZSk7XG4gICNvdmVybGF5ID0gaW5qZWN0KFl2Y092ZXJsYXlTZXJ2aWNlKTtcbiAgcHJpdmF0ZSB0cmFuc2xhdGUgPSBpbmplY3QoVHJhbnNsYXRlU2VydmljZSk7XG5cbiAgaWQgPSAneXV2LmJhc2UubWFuYWdlLWZsYXZvcic7XG4gIGxhYmVsID0gdGhpcy50cmFuc2xhdGUuaW5zdGFudCgneXV2LnNoZWxsLmFjdGlvbi5tYW5hZ2UtZmxhdm9yLmxhYmVsJyk7XG4gIGRlc2NyaXB0aW9uID0gdGhpcy50cmFuc2xhdGUuaW5zdGFudCgneXV2LnNoZWxsLmFjdGlvbi5tYW5hZ2UtZmxhdm9yLmRlc2NyaXB0aW9uJyk7XG4gIHByaW9yaXR5ID0gODtcbiAgaWNvbiA9IEFDVElPTl9JQ09OLm1hbmFnZUZsYXZvcjtcbiAgZ3JvdXAgPSAnY29tbW9uJztcbiAgcmFuZ2UgPSBTZWxlY3Rpb25SYW5nZS5TSU5HTEVfU0VMRUNUO1xuICBzdXBwb3J0czogQWN0aW9uU3VwcG9ydCA9IHtcbiAgICBwYXR0ZXJuOiAnKidcbiAgfTtcblxuICBpc0V4ZWN1dGFibGUoaXRlbXM6IERtc09iamVjdFtdKSB7XG4gICAgY29uc3QgaXRlbSA9IGl0ZW1zWzBdO1xuICAgIHJldHVybiBvZihpdGVtICYmICEhaXRlbS5wZXJtaXNzaW9ucz8ud3JpdGVJbmRleERhdGEgJiYgISFpdGVtLmNvbnRlbnQgJiYgdGhpcy4jc2hlbGwuZ2V0QXBwbGljYWJsZU9iamVjdEZsYXZvcnMoaXRlbS5jb250ZW50Lm1pbWVUeXBlKS5sZW5ndGggPiAwKTtcbiAgfVxuXG4gIHJ1bihpdGVtczogRG1zT2JqZWN0W10pOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICB0aGlzLiNvdmVybGF5Lm9wZW4oTWFuYWdlRmxhdm9yc0NvbXBvbmVudCwgaXRlbXNbMF0sIHtcbiAgICAgIHdpZHRoOiAnNDAwcHgnLFxuICAgICAgbWF4V2lkdGg6ICc5MHZ3JyxcbiAgICB9KTtcbiAgICByZXR1cm4gb2YodHJ1ZSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,69 @@
1
+ import { Component, inject, signal } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { YvcOverlayRef } from '@yuuvis/components/overlay';
4
+ import { DmsService, TranslateModule } from '@yuuvis/client-core';
5
+ import { ShellService } from '@yuuvis/client-shell-core';
6
+ import { FlavorChipComponent } from '@yuuvis/client-framework/object-flavor';
7
+ import { ListComponent, ListItemDirective } from '@yuuvis/client-framework/list';
8
+ import { BusyOverlayDirective } from '@yuuvis/client-framework/common';
9
+ import { tap } from 'rxjs';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "@yuuvis/client-core";
12
+ export class ManageFlavorsComponent {
13
+ constructor() {
14
+ this.#overlayRef = inject(YvcOverlayRef);
15
+ this.#shell = inject(ShellService);
16
+ this.#dmsService = inject(DmsService);
17
+ this.item = this.#overlayRef.data;
18
+ this.busy = signal(false);
19
+ this.appliedFlavors = [];
20
+ this.applicableFlavors = [];
21
+ }
22
+ #overlayRef;
23
+ #shell;
24
+ #dmsService;
25
+ applyFlavor(flavor) {
26
+ this.busy.set(true);
27
+ this.#shell.triggerApplyObjectFlavor(this.item, flavor).subscribe(() => {
28
+ this.#overlayRef.close();
29
+ this.busy.set(false);
30
+ });
31
+ }
32
+ removeFlavor(flavor) {
33
+ this.busy.set(true);
34
+ this.#shell.removeObjectFlavor(this.item, flavor).subscribe(() => {
35
+ this.#overlayRef.close();
36
+ this.busy.set(false);
37
+ });
38
+ }
39
+ #refreshDmsObject() {
40
+ this.busy.set(true);
41
+ this.#dmsService
42
+ .getDmsObject(this.item.id)
43
+ .pipe(tap((dmsObject) => {
44
+ this.item = dmsObject;
45
+ this.#getAppliedFlavors(this.item);
46
+ }))
47
+ .subscribe()
48
+ .add(() => this.busy.set(false));
49
+ }
50
+ #getAppliedFlavors(dmsObject) {
51
+ const res = this.#shell.getAppliedObjectFlavors(dmsObject);
52
+ this.appliedFlavors = res.applied;
53
+ this.applicableFlavors = res.applicable;
54
+ }
55
+ cancel() {
56
+ if (this.#overlayRef)
57
+ this.#overlayRef.close();
58
+ }
59
+ ngOnInit() {
60
+ this.#refreshDmsObject();
61
+ }
62
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ManageFlavorsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
63
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ManageFlavorsComponent, isStandalone: true, selector: "yuv-manage-flavors", ngImport: i0, template: "<header>\n <h2>{{ 'yuv.shell.action.manage-flavors.title' | translate }}</h2>\n <p>{{ 'yuv.shell.action.manage-flavors.text' | translate }}</p>\n</header>\n<main [yuvBusyOverlay]=\"busy()\">\n @if (appliedFlavors.length) {\n <h3>{{ 'yuv.shell.action.manage-flavors.applied.headline' | translate }}</h3>\n <ul>\n @for (f of appliedFlavors; track $index) {\n <li>\n <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n <button class=\"secondary\" (click)=\"removeFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applicable.button.remove' | translate }}</button>\n </li>\n }\n </ul>\n }\n @if (applicableFlavors.length) {\n <h3>{{ 'yuv.shell.action.manage-flavors.applicable.headline' | translate }}</h3>\n <ul>\n @for (f of applicableFlavors; track $index) {\n <li>\n <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n <button class=\"secondary\" (click)=\"applyFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applied.button.apply' | translate }}</button>\n </li>\n }\n </ul>\n }\n</main>\n<footer>\n <button class=\"secondary\" (click)=\"cancel()\">{{ 'yuv.shell.action.manage-flavors.button.cancel' | translate }}</button>\n</footer>\n", styles: [":host{display:flex;flex-flow:column;padding:var(--app-pane-padding)}:host header{flex:0 0 auto}:host header h2{padding:0;font-weight:400;font-size:var(--font-headline);margin:0 0 var(--app-pane-padding) 0}:host header p{max-width:42ch;line-height:1.3em}:host main{flex:1;overflow-y:auto}:host main yuv-flavor-chip{border-color:transparent;flex:1}:host main ul{padding:0;margin:0;list-style-type:none}:host main li{display:flex;gap:calc(var(--app-pane-padding) / 2);align-items:center;border:1px solid var(--panel-divider-color);margin-block-end:2px;padding:calc(var(--app-pane-padding) / 4)}:host main li button{padding:calc(var(--app-pane-padding) / 4) calc(var(--app-pane-padding) / 2)}:host footer{flex:0 0 auto;margin-block-start:var(--app-pane-padding)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: BusyOverlayDirective, selector: "[yuvBusyOverlay]", inputs: ["yuvBusyOverlay"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }, { kind: "directive", type: ListItemDirective, selector: "[yuvListItem]", inputs: ["disabled", "active", "selected"] }, { kind: "component", type: FlavorChipComponent, selector: "yuv-flavor-chip", inputs: ["flavor", "enableRemove", "enableDescription"], outputs: ["flavorRemove"] }] }); }
64
+ }
65
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ManageFlavorsComponent, decorators: [{
66
+ type: Component,
67
+ args: [{ selector: 'yuv-manage-flavors', standalone: true, imports: [CommonModule, ListComponent, BusyOverlayDirective, TranslateModule, ListItemDirective, FlavorChipComponent], template: "<header>\n <h2>{{ 'yuv.shell.action.manage-flavors.title' | translate }}</h2>\n <p>{{ 'yuv.shell.action.manage-flavors.text' | translate }}</p>\n</header>\n<main [yuvBusyOverlay]=\"busy()\">\n @if (appliedFlavors.length) {\n <h3>{{ 'yuv.shell.action.manage-flavors.applied.headline' | translate }}</h3>\n <ul>\n @for (f of appliedFlavors; track $index) {\n <li>\n <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n <button class=\"secondary\" (click)=\"removeFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applicable.button.remove' | translate }}</button>\n </li>\n }\n </ul>\n }\n @if (applicableFlavors.length) {\n <h3>{{ 'yuv.shell.action.manage-flavors.applicable.headline' | translate }}</h3>\n <ul>\n @for (f of applicableFlavors; track $index) {\n <li>\n <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n <button class=\"secondary\" (click)=\"applyFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applied.button.apply' | translate }}</button>\n </li>\n }\n </ul>\n }\n</main>\n<footer>\n <button class=\"secondary\" (click)=\"cancel()\">{{ 'yuv.shell.action.manage-flavors.button.cancel' | translate }}</button>\n</footer>\n", styles: [":host{display:flex;flex-flow:column;padding:var(--app-pane-padding)}:host header{flex:0 0 auto}:host header h2{padding:0;font-weight:400;font-size:var(--font-headline);margin:0 0 var(--app-pane-padding) 0}:host header p{max-width:42ch;line-height:1.3em}:host main{flex:1;overflow-y:auto}:host main yuv-flavor-chip{border-color:transparent;flex:1}:host main ul{padding:0;margin:0;list-style-type:none}:host main li{display:flex;gap:calc(var(--app-pane-padding) / 2);align-items:center;border:1px solid var(--panel-divider-color);margin-block-end:2px;padding:calc(var(--app-pane-padding) / 4)}:host main li button{padding:calc(var(--app-pane-padding) / 4) calc(var(--app-pane-padding) / 2)}:host footer{flex:0 0 auto;margin-block-start:var(--app-pane-padding)}\n"] }]
68
+ }] });
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"manage-flavors.component.js","sourceRoot":"","sources":["../../../../../../../../libs/yuuvis/client-shell/src/lib/actions/manage-flavors/manage-flavors.component.ts","../../../../../../../../libs/yuuvis/client-shell/src/lib/actions/manage-flavors/manage-flavors.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, OnInit, signal } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { YvcOverlayRef } from '@yuuvis/components/overlay';\nimport { DmsObject, DmsService, TranslateModule, TranslateService } from '@yuuvis/client-core';\nimport { ObjectFlavor, ShellService } from '@yuuvis/client-shell-core';\nimport { FlavorChipComponent } from '@yuuvis/client-framework/object-flavor';\nimport { ListComponent, ListItemDirective } from '@yuuvis/client-framework/list';\nimport { BusyOverlayDirective } from '@yuuvis/client-framework/common';\nimport { Observable, tap } from 'rxjs';\n\n@Component({\n  selector: 'yuv-manage-flavors',\n  standalone: true,\n  imports: [CommonModule, ListComponent, BusyOverlayDirective, TranslateModule, ListItemDirective, FlavorChipComponent],\n  templateUrl: './manage-flavors.component.html',\n  styleUrl: './manage-flavors.component.scss'\n})\nexport class ManageFlavorsComponent implements OnInit {\n  #overlayRef = inject(YvcOverlayRef);\n  #shell = inject(ShellService);\n  #dmsService = inject(DmsService);\n  item: DmsObject = this.#overlayRef.data;\n\n  busy = signal<boolean>(false);\n\n  appliedFlavors: ObjectFlavor[] = [];\n  applicableFlavors: ObjectFlavor[] = [];\n\n  applyFlavor(flavor: ObjectFlavor) {\n    this.busy.set(true);\n    this.#shell.triggerApplyObjectFlavor(this.item, flavor).subscribe(() => {\n      this.#overlayRef.close();\n      this.busy.set(false);\n    });\n  }\n\n  removeFlavor(flavor: ObjectFlavor) {\n    this.busy.set(true);\n    this.#shell.removeObjectFlavor(this.item, flavor).subscribe(() => {\n      this.#overlayRef.close();\n      this.busy.set(false);\n    });\n  }\n\n  #refreshDmsObject() {\n    this.busy.set(true);\n    this.#dmsService\n      .getDmsObject(this.item.id)\n      .pipe(\n        tap((dmsObject) => {\n          this.item = dmsObject;\n          this.#getAppliedFlavors(this.item);\n        })\n      )\n      .subscribe()\n      .add(() => this.busy.set(false));\n  }\n\n  #getAppliedFlavors(dmsObject: DmsObject) {\n    const res = this.#shell.getAppliedObjectFlavors(dmsObject);\n    this.appliedFlavors = res.applied;\n    this.applicableFlavors = res.applicable;\n  }\n\n  cancel() {\n    if (this.#overlayRef) this.#overlayRef.close();\n  }\n\n  ngOnInit() {\n    this.#refreshDmsObject();\n  }\n}\n","<header>\n  <h2>{{ 'yuv.shell.action.manage-flavors.title' | translate }}</h2>\n  <p>{{ 'yuv.shell.action.manage-flavors.text' | translate }}</p>\n</header>\n<main [yuvBusyOverlay]=\"busy()\">\n  @if (appliedFlavors.length) {\n    <h3>{{ 'yuv.shell.action.manage-flavors.applied.headline' | translate }}</h3>\n    <ul>\n      @for (f of appliedFlavors; track $index) {\n        <li>\n          <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n          <button class=\"secondary\" (click)=\"removeFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applicable.button.remove' | translate }}</button>\n        </li>\n      }\n    </ul>\n  }\n  @if (applicableFlavors.length) {\n    <h3>{{ 'yuv.shell.action.manage-flavors.applicable.headline' | translate }}</h3>\n    <ul>\n      @for (f of applicableFlavors; track $index) {\n        <li>\n          <yuv-flavor-chip [flavor]=\"f\" yuvListItem></yuv-flavor-chip>\n          <button class=\"secondary\" (click)=\"applyFlavor(f)\">{{ 'yuv.shell.action.manage-flavors.applied.button.apply' | translate }}</button>\n        </li>\n      }\n    </ul>\n  }\n</main>\n<footer>\n  <button class=\"secondary\" (click)=\"cancel()\">{{ 'yuv.shell.action.manage-flavors.button.cancel' | translate }}</button>\n</footer>\n"]}
@@ -0,0 +1,191 @@
1
+ import { AsyncPipe, NgIf } from '@angular/common';
2
+ import { Component, HostListener, effect, inject, input, signal } from '@angular/core';
3
+ import { toSignal } from '@angular/core/rxjs-interop';
4
+ import { NavigationEnd, Router, RouterModule } from '@angular/router';
5
+ import { SwPush } from '@angular/service-worker';
6
+ import { AuthService, DeviceService, TranslateModule, TranslateService, UserRoles, UserService, Utils } from '@yuuvis/client-core';
7
+ import { MetadataDefaultTemplatesComponent } from '@yuuvis/client-framework/metadata-form';
8
+ import { CommandPaletteService, ShellNotificationsService, ShellService } from '@yuuvis/client-shell-core';
9
+ import { YvcIconModule } from '@yuuvis/components/icon';
10
+ import { of } from 'rxjs';
11
+ import { catchError, filter, map, switchMap, tap } from 'rxjs/operators';
12
+ import { clientShellRoutes } from './lib.routes';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "@yuuvis/client-core";
15
+ import * as i2 from "@angular/router";
16
+ import * as i3 from "@yuuvis/components/icon";
17
+ export class ClientShellComponent {
18
+ #shell;
19
+ #device;
20
+ #swPush;
21
+ onFocusChange(event) {
22
+ // console.log('focused: ', document.activeElement);
23
+ }
24
+ onDragOver(event) {
25
+ event.stopPropagation();
26
+ event.preventDefault();
27
+ this.showUploadOverlay = !!this._dragContainsFiles(event);
28
+ }
29
+ #appsEffect;
30
+ #shellConfigEffect;
31
+ constructor() {
32
+ this.router = inject(Router);
33
+ this.auth = inject(AuthService);
34
+ this.userService = inject(UserService);
35
+ this.#shell = inject(ShellService);
36
+ this.shellNotifications = inject(ShellNotificationsService);
37
+ this.translate = inject(TranslateService);
38
+ this.commandPalette = inject(CommandPaletteService);
39
+ this.#device = inject(DeviceService);
40
+ this.#swPush = inject(SwPush);
41
+ this.APP_LOGOUT_EVENT_KEY = 'yuv.app.event.logout';
42
+ this._levels = {
43
+ info: 0,
44
+ success: 1,
45
+ warning: 2,
46
+ alert: 3
47
+ };
48
+ this.showUploadOverlay = false;
49
+ this.busy$ = this.#shell.isBusy$;
50
+ this.showNotifications = signal(false);
51
+ this.newNotifications = toSignal(this.shellNotifications.shellNotifications$.pipe(tap((n) => this.showNotifications.set(n.length > 0)), map((notifications) => {
52
+ let maxLevel = 'info';
53
+ const count = notifications.filter((n) => !n.seen).length;
54
+ notifications.forEach((n) => (maxLevel = n.level && this._levels[n.level] > this._levels[maxLevel] ? n.level : maxLevel));
55
+ return count > 0 ? { maxLevel, count } : undefined;
56
+ })));
57
+ this.apps = input.required();
58
+ this.#appsEffect = effect(() => this.#shell.setAppBaseRoutes(this.apps()));
59
+ this.checkedForInitialRoute = false;
60
+ this.enableTenantSwitch = false;
61
+ this.config = input();
62
+ this.#shellConfigEffect = effect(() => {
63
+ const cfg = this.config();
64
+ if (cfg) {
65
+ this.#shell.setShellConfig(cfg);
66
+ }
67
+ }, {
68
+ allowSignalWrites: true
69
+ });
70
+ this.shellConfig = this.#shell.shellConfig;
71
+ this.translate.onLangChange.subscribe(() => this._setCommands(true));
72
+ // this.router.events.pipe(
73
+ // // filter(e => e instanceof NavigationStart)
74
+ // ).subscribe((e) => {
75
+ // console.log(e);
76
+ // });
77
+ this.router.events
78
+ .pipe(filter((e) => e instanceof NavigationEnd), map((e) => e))
79
+ .subscribe((e) => this._processRouterNavigationEnd(e));
80
+ this.#device.init();
81
+ this.userService.user$.subscribe((user) => {
82
+ if (user) {
83
+ this.checkedForInitialRoute = !(!this.user || this.user.id !== user.id);
84
+ this.enableTenantSwitch = user.authorities.includes(UserRoles.MULTI_TENANT);
85
+ }
86
+ this.user = user;
87
+ });
88
+ window.addEventListener('storage', (evt) => {
89
+ if (evt.key === this.APP_LOGOUT_EVENT_KEY) {
90
+ this.appLogout(true);
91
+ }
92
+ });
93
+ }
94
+ openNotifications() {
95
+ this.router.navigate([{}]);
96
+ }
97
+ // getNotifications(id: string) {
98
+ // return this.shellNotifications.getNotifications(id);
99
+ // }
100
+ _dragContainsFiles(event) {
101
+ // do not allow to drop files/images from iframes
102
+ if (event.relatedTarget?.tagName.toLowerCase() === 'iframe')
103
+ return 0;
104
+ return event.dataTransfer ? Array.from(event.dataTransfer.items || []).filter((i) => i.kind === 'file' && i.type).length : 0;
105
+ }
106
+ _setCommands(update) {
107
+ const commands = this.apps().map((a, i) => ({
108
+ id: `nav.app.${i}`,
109
+ label: this.translate.instant('yuv.shell.cmd.app.open', { title: a.title }),
110
+ callback: () => this.router.navigate([a.path])
111
+ }));
112
+ commands.push({
113
+ id: `nav.shell.settings`,
114
+ label: this.translate.instant('yuv.shell.cmd.open.settings'),
115
+ callback: () => this.router.navigate(['settings'])
116
+ });
117
+ commands.push({
118
+ id: `nav.shell.logout`,
119
+ label: this.translate.instant('yuv.shell.cmd.logout'),
120
+ callback: () => this.appLogout()
121
+ });
122
+ if (update)
123
+ this.commandPalette.updateCommands(commands);
124
+ else
125
+ this.commandPalette.registerCommands(commands);
126
+ }
127
+ requestSubscription() {
128
+ if (!this.#swPush.isEnabled) {
129
+ console.log('Notification is not enabled.');
130
+ return;
131
+ }
132
+ this.#swPush.messages
133
+ .pipe(tap((msg) => console.log({ msg })), catchError((error) => {
134
+ console.log({ error });
135
+ return of(null);
136
+ }))
137
+ .subscribe();
138
+ }
139
+ appLogout(triggeredFromOtherTab) {
140
+ if (!triggeredFromOtherTab) {
141
+ // send storage event to logout all other open tabs
142
+ window.localStorage.setItem(this.APP_LOGOUT_EVENT_KEY, `${Date.now()}`);
143
+ }
144
+ this.userService.logout('');
145
+ }
146
+ _processRouterNavigationEnd(e) {
147
+ if (!this.checkedForInitialRoute) {
148
+ this.checkedForInitialRoute = true;
149
+ // redirect to the page the user logged out from the last time
150
+ // but only if current route is not a deep link
151
+ const ignoreRoutes = ['', 'dashboard', 'index.html'].map((s) => `${Utils.getBaseHref()}${s}`.replace('//', '/'));
152
+ const currentRoute = this.routeWithBaseHref(this.router.routerState.snapshot.url);
153
+ if (this.userService.getCurrentUser() && !ignoreRoutes.includes(currentRoute)) {
154
+ // get persisted routes to decide where to redirect the logged in user to
155
+ this.auth
156
+ .getInitialRequestUri()
157
+ .pipe(switchMap((res) => this.auth.resetInitialRequestUri().pipe(map((_) => res))))
158
+ .subscribe((res) => {
159
+ const loginRes = res && !ignoreRoutes.includes(res.uri) ? res : null;
160
+ if (loginRes)
161
+ this.router.navigateByUrl(loginRes.uri);
162
+ });
163
+ }
164
+ }
165
+ }
166
+ routeWithBaseHref(r) {
167
+ return `${Utils.getBaseHref()}${r}`.replace('//', '/');
168
+ }
169
+ ngOnInit() {
170
+ this.#shell._init();
171
+ clientShellRoutes.forEach((route) => {
172
+ this.router.config.push(route);
173
+ });
174
+ this.router.resetConfig(this.router.config);
175
+ this._setCommands();
176
+ this.requestSubscription();
177
+ }
178
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ClientShellComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
179
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ClientShellComponent, isStandalone: true, selector: "yuv-client-shell", inputs: { apps: { classPropertyName: "apps", publicName: "apps", isSignal: true, isRequired: true, transformFunction: null }, config: { classPropertyName: "config", publicName: "config", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "document:focusin": "onFocusChange($event)", "dragover": "onDragOver($event)" } }, ngImport: i0, template: "<yuv-metadata-default-templates></yuv-metadata-default-templates>\n<!-- <yuv-focus-indicator /> -->\n\n<!-- gloabl busy indicator -->\n@if (busy$ | async) {\n <div class=\"yuv-loader-linear\"></div>\n}\n<nav class=\"apps\" [inert]=\"asideOutlet.isActivated\" [attr.aria-label]=\"'yuv.shell.naviagtion.aria.label' | translate\">\n <div class=\"app-icon\">\n <button routerLink=\"/\" routerLinkActive=\"active\" [attr.aria-label]=\"'yuv.shell.logo.aria.label' | translate\">\n <yvc-icon [svg]=\"shellConfig().icons!.appIcon!\"></yvc-icon>\n </button>\n </div>\n <ul class=\"apps\">\n @for (a of apps(); track a.path) {\n <li>\n <button routerLinkActive=\"active\" [routerLink]=\"a.path\" title=\"{{ a.title }}\">\n <yvc-icon [svg]=\"a.icon\" title=\"{{ a.title }}\" *ngIf=\"a.icon\"></yvc-icon>\n <!-- @if (getNotifications(a.id)) {\n <div [ngClass]=\"'badge ' + getNotifications(a.id).maxLevel\">{{ getNotifications(a.id).count }}</div>\n } -->\n </button>\n </li>\n }\n </ul>\n\n <section class=\"actions\">\n @if (showNotifications()) {\n <button [routerLink]=\"[{ outlets: { aside: 'notifications' } }]\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.notifications.title' | translate }}\">\n <yvc-icon [svg]=\"shellConfig().icons!.notifications!\"></yvc-icon>\n @if (newNotifications(); as note) {\n <div class=\"badge {{ note.maxLevel }}\">{{ note.count }}</div>\n }\n </button>\n }\n <button [routerLink]=\"['/settings']\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.settings.title' | translate }}\">\n <yvc-icon [svg]=\"shellConfig().icons!.settings!\"></yvc-icon>\n </button>\n <button (click)=\"appLogout()\" title=\"{{ 'yuv.shell.cmd.logout' | translate }}\"><yvc-icon [svg]=\"shellConfig().icons!.logout!\"></yvc-icon></button>\n </section>\n</nav>\n<main id=\"main-shell_content\" aria-label=\"main shell content\" [inert]=\"asideOutlet.isActivated\">\n <router-outlet></router-outlet>\n</main>\n\n<!-- outlet for aside modals like notifications -->\n<div class=\"asideOutlet\" [hidden]=\"!asideOutlet.isActivated\">\n <router-outlet name=\"aside\" #asideOutlet=\"outlet\"></router-outlet>\n</div>\n\n<div id=\"fi\" inert></div>\n", styles: [":host{--client-shell-background-color: var(--main-background);--client-shell-bar-background-color: var(--panel-background);--client-shell-bar-icon-size: 24px;--client-shell-bar-button-padding: calc(var(--app-pane-padding) * .75);--client-shell-bar-width: calc(var(--client-shell-bar-icon-size) + var(--client-shell-bar-button-padding) * 2 + 1px);position:absolute;inset:0;overflow:hidden;display:flex;background-color:var(--client-shell-background-color)}:host .yuv-loader-linear{position:absolute}:host nav{background-color:var(--client-shell-bar-background-color);height:100%;overflow-y:auto;border-inline-end:1px solid var(--panel-divider-color);flex:0 0 auto;display:var(--nav-display, grid);grid-template-rows:auto 1fr auto;grid-template-columns:1fr;grid-template-areas:\"appIcon\" \"apps\" \"actions\"}:host nav button{padding:var(--client-shell-bar-button-padding);border-radius:0;border:0}:host nav button yvc-icon{--icon-size: var(--client-shell-bar-icon-size)}:host nav button.active{color:var(--color-accent)}:host nav .app-icon{grid-area:appIcon}:host nav ul.apps{grid-area:apps;list-style:none;margin:0;padding:0;overflow-y:auto}:host nav ul.apps button{position:relative}:host nav ul.apps button .badge{--badge-color: var(--color-accent);--badge-color-tone: var(--color-accent-tone);position:absolute;background-color:var(--badge-color);color:var(--badge-color-tone);font-size:10px;display:block;padding:1px 2px;border-radius:.25em;line-height:1em;top:calc(var(--app-pane-padding) / 4);right:calc(var(--app-pane-padding) / 4)}:host nav ul.apps button .badge.success{--badge-color: var(--color-success);--badge-color-tone: #fff}:host nav ul.apps button .badge.warning{--badge-color: var(--color-warning);--badge-color-tone: #fff}:host nav ul.apps button .badge.alert{--badge-color: var(--color-error);--badge-color-tone: #fff}:host nav ul.apps li a{display:block;padding:var(--app-pane-padding)}:host nav section.actions{grid-area:actions}:host nav section.actions button{position:relative}:host nav section.actions button .badge{--badge-color: var(--color-accent);--badge-color-tone: var(--color-accent-tone);position:absolute;background-color:var(--badge-color);color:var(--badge-color-tone);font-size:10px;display:block;padding:1px 2px;border-radius:.25em;line-height:1em;top:calc(var(--app-pane-padding) / 4);right:calc(var(--app-pane-padding) / 4)}:host nav section.actions button .badge.success{--badge-color: var(--color-success);--badge-color-tone: #fff}:host nav section.actions button .badge.warning{--badge-color: var(--color-warning);--badge-color-tone: #fff}:host nav section.actions button .badge.alert{--badge-color: var(--color-error);--badge-color-tone: #fff}:host main{flex:1;color:var(--text-color-body)}:host .asideOutlet{position:absolute;inset:0;padding-inline-start:var(--client-shell-bar-width)}:host-context([data-screen-size=s][data-screen-orientation=portrait]){flex-flow:column-reverse}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav{height:auto;grid-template-rows:1fr;grid-template-columns:auto 1fr auto;grid-template-areas:\"appIcon apps actions\";border:0;border-block-start:1px solid var(--panel-divider-color)}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav .app-icon{display:none}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav ul.apps,:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav section.actions{display:flex}:host-context([data-screen-size=s][data-screen-orientation=portrait]) main{overflow:hidden}:host-context([data-screen-size=s][data-screen-orientation=portrait]) .asideOutlet{padding-inline-start:0}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: RouterModule }, { kind: "directive", type: i2.RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }, { kind: "directive", type: i2.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i2.RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "ngmodule", type: YvcIconModule }, { kind: "component", type: i3.Icon, selector: "yvc-icon", inputs: ["label", "svg", "svgSrc"] }, { kind: "component", type: MetadataDefaultTemplatesComponent, selector: "yuv-metadata-default-templates" }] }); }
180
+ }
181
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ClientShellComponent, decorators: [{
182
+ type: Component,
183
+ args: [{ selector: 'yuv-client-shell', standalone: true, imports: [NgIf, AsyncPipe, TranslateModule, RouterModule, YvcIconModule, MetadataDefaultTemplatesComponent], template: "<yuv-metadata-default-templates></yuv-metadata-default-templates>\n<!-- <yuv-focus-indicator /> -->\n\n<!-- gloabl busy indicator -->\n@if (busy$ | async) {\n <div class=\"yuv-loader-linear\"></div>\n}\n<nav class=\"apps\" [inert]=\"asideOutlet.isActivated\" [attr.aria-label]=\"'yuv.shell.naviagtion.aria.label' | translate\">\n <div class=\"app-icon\">\n <button routerLink=\"/\" routerLinkActive=\"active\" [attr.aria-label]=\"'yuv.shell.logo.aria.label' | translate\">\n <yvc-icon [svg]=\"shellConfig().icons!.appIcon!\"></yvc-icon>\n </button>\n </div>\n <ul class=\"apps\">\n @for (a of apps(); track a.path) {\n <li>\n <button routerLinkActive=\"active\" [routerLink]=\"a.path\" title=\"{{ a.title }}\">\n <yvc-icon [svg]=\"a.icon\" title=\"{{ a.title }}\" *ngIf=\"a.icon\"></yvc-icon>\n <!-- @if (getNotifications(a.id)) {\n <div [ngClass]=\"'badge ' + getNotifications(a.id).maxLevel\">{{ getNotifications(a.id).count }}</div>\n } -->\n </button>\n </li>\n }\n </ul>\n\n <section class=\"actions\">\n @if (showNotifications()) {\n <button [routerLink]=\"[{ outlets: { aside: 'notifications' } }]\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.notifications.title' | translate }}\">\n <yvc-icon [svg]=\"shellConfig().icons!.notifications!\"></yvc-icon>\n @if (newNotifications(); as note) {\n <div class=\"badge {{ note.maxLevel }}\">{{ note.count }}</div>\n }\n </button>\n }\n <button [routerLink]=\"['/settings']\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.settings.title' | translate }}\">\n <yvc-icon [svg]=\"shellConfig().icons!.settings!\"></yvc-icon>\n </button>\n <button (click)=\"appLogout()\" title=\"{{ 'yuv.shell.cmd.logout' | translate }}\"><yvc-icon [svg]=\"shellConfig().icons!.logout!\"></yvc-icon></button>\n </section>\n</nav>\n<main id=\"main-shell_content\" aria-label=\"main shell content\" [inert]=\"asideOutlet.isActivated\">\n <router-outlet></router-outlet>\n</main>\n\n<!-- outlet for aside modals like notifications -->\n<div class=\"asideOutlet\" [hidden]=\"!asideOutlet.isActivated\">\n <router-outlet name=\"aside\" #asideOutlet=\"outlet\"></router-outlet>\n</div>\n\n<div id=\"fi\" inert></div>\n", styles: [":host{--client-shell-background-color: var(--main-background);--client-shell-bar-background-color: var(--panel-background);--client-shell-bar-icon-size: 24px;--client-shell-bar-button-padding: calc(var(--app-pane-padding) * .75);--client-shell-bar-width: calc(var(--client-shell-bar-icon-size) + var(--client-shell-bar-button-padding) * 2 + 1px);position:absolute;inset:0;overflow:hidden;display:flex;background-color:var(--client-shell-background-color)}:host .yuv-loader-linear{position:absolute}:host nav{background-color:var(--client-shell-bar-background-color);height:100%;overflow-y:auto;border-inline-end:1px solid var(--panel-divider-color);flex:0 0 auto;display:var(--nav-display, grid);grid-template-rows:auto 1fr auto;grid-template-columns:1fr;grid-template-areas:\"appIcon\" \"apps\" \"actions\"}:host nav button{padding:var(--client-shell-bar-button-padding);border-radius:0;border:0}:host nav button yvc-icon{--icon-size: var(--client-shell-bar-icon-size)}:host nav button.active{color:var(--color-accent)}:host nav .app-icon{grid-area:appIcon}:host nav ul.apps{grid-area:apps;list-style:none;margin:0;padding:0;overflow-y:auto}:host nav ul.apps button{position:relative}:host nav ul.apps button .badge{--badge-color: var(--color-accent);--badge-color-tone: var(--color-accent-tone);position:absolute;background-color:var(--badge-color);color:var(--badge-color-tone);font-size:10px;display:block;padding:1px 2px;border-radius:.25em;line-height:1em;top:calc(var(--app-pane-padding) / 4);right:calc(var(--app-pane-padding) / 4)}:host nav ul.apps button .badge.success{--badge-color: var(--color-success);--badge-color-tone: #fff}:host nav ul.apps button .badge.warning{--badge-color: var(--color-warning);--badge-color-tone: #fff}:host nav ul.apps button .badge.alert{--badge-color: var(--color-error);--badge-color-tone: #fff}:host nav ul.apps li a{display:block;padding:var(--app-pane-padding)}:host nav section.actions{grid-area:actions}:host nav section.actions button{position:relative}:host nav section.actions button .badge{--badge-color: var(--color-accent);--badge-color-tone: var(--color-accent-tone);position:absolute;background-color:var(--badge-color);color:var(--badge-color-tone);font-size:10px;display:block;padding:1px 2px;border-radius:.25em;line-height:1em;top:calc(var(--app-pane-padding) / 4);right:calc(var(--app-pane-padding) / 4)}:host nav section.actions button .badge.success{--badge-color: var(--color-success);--badge-color-tone: #fff}:host nav section.actions button .badge.warning{--badge-color: var(--color-warning);--badge-color-tone: #fff}:host nav section.actions button .badge.alert{--badge-color: var(--color-error);--badge-color-tone: #fff}:host main{flex:1;color:var(--text-color-body)}:host .asideOutlet{position:absolute;inset:0;padding-inline-start:var(--client-shell-bar-width)}:host-context([data-screen-size=s][data-screen-orientation=portrait]){flex-flow:column-reverse}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav{height:auto;grid-template-rows:1fr;grid-template-columns:auto 1fr auto;grid-template-areas:\"appIcon apps actions\";border:0;border-block-start:1px solid var(--panel-divider-color)}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav .app-icon{display:none}:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav ul.apps,:host-context([data-screen-size=s][data-screen-orientation=portrait]) nav section.actions{display:flex}:host-context([data-screen-size=s][data-screen-orientation=portrait]) main{overflow:hidden}:host-context([data-screen-size=s][data-screen-orientation=portrait]) .asideOutlet{padding-inline-start:0}\n"] }]
184
+ }], ctorParameters: () => [], propDecorators: { onFocusChange: [{
185
+ type: HostListener,
186
+ args: ['document:focusin', ['$event']]
187
+ }], onDragOver: [{
188
+ type: HostListener,
189
+ args: ['dragover', ['$event']]
190
+ }] } });
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-shell.component.js","sourceRoot":"","sources":["../../../../../../libs/yuuvis/client-shell/src/lib/client-shell.component.ts","../../../../../../libs/yuuvis/client-shell/src/lib/client-shell.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AsyncPipe, NgIf } from '@angular/common';\nimport { Component, HostListener, OnInit, Signal, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { NavigationEnd, Route, Router, RouterModule } from '@angular/router';\nimport { SwPush } from '@angular/service-worker';\nimport { AuthService, DeviceService, TranslateModule, TranslateService, UserRoles, UserService, Utils, YuvUser } from '@yuuvis/client-core';\nimport { MetadataDefaultTemplatesComponent } from '@yuuvis/client-framework/metadata-form';\nimport {\n  App,\n  ClientShellConfig,\n  CommandPaletteCommand,\n  CommandPaletteService,\n  ShellNotificationLevel,\n  ShellNotificationsService,\n  ShellService\n} from '@yuuvis/client-shell-core';\nimport { YvcIconModule } from '@yuuvis/components/icon';\nimport { of } from 'rxjs';\nimport { catchError, filter, map, switchMap, tap } from 'rxjs/operators';\nimport { clientShellRoutes } from './lib.routes';\n\n@Component({\n  selector: 'yuv-client-shell',\n  standalone: true,\n  imports: [NgIf, AsyncPipe, TranslateModule, RouterModule, YvcIconModule, MetadataDefaultTemplatesComponent],\n  templateUrl: './client-shell.component.html',\n  styleUrls: ['./client-shell.component.scss']\n})\nexport class ClientShellComponent implements OnInit {\n  private router = inject(Router);\n  private auth = inject(AuthService);\n  private userService = inject(UserService);\n  #shell = inject(ShellService);\n  private shellNotifications = inject(ShellNotificationsService);\n  private translate = inject(TranslateService);\n  private commandPalette = inject(CommandPaletteService);\n  readonly #device = inject(DeviceService);\n  readonly #swPush = inject(SwPush);\n\n  private APP_LOGOUT_EVENT_KEY = 'yuv.app.event.logout';\n\n  private _levels: Record<ShellNotificationLevel, number> = {\n    info: 0,\n    success: 1,\n    warning: 2,\n    alert: 3\n  };\n\n  showUploadOverlay = false;\n  busy$ = this.#shell.isBusy$;\n  context?: string;\n\n  showNotifications = signal<boolean>(false);\n  newNotifications: Signal<\n    | {\n        count: number;\n        maxLevel: ShellNotificationLevel;\n      }\n    | undefined\n  > = toSignal(\n    this.shellNotifications.shellNotifications$.pipe(\n      tap((n) => this.showNotifications.set(n.length > 0)),\n      map((notifications) => {\n        let maxLevel: ShellNotificationLevel = 'info';\n        const count = notifications.filter((n) => !n.seen).length;\n        notifications.forEach((n) => (maxLevel = n.level && this._levels[n.level] > this._levels[maxLevel] ? n.level : maxLevel));\n        return count > 0 ? { maxLevel, count } : undefined;\n      })\n    )\n  );\n\n  @HostListener('document:focusin', ['$event'])\n  onFocusChange(event: FocusEvent) {\n    // console.log('focused: ', document.activeElement);\n  }\n\n  @HostListener('dragover', ['$event'])\n  onDragOver(event: DragEvent) {\n    event.stopPropagation();\n    event.preventDefault();\n\n    this.showUploadOverlay = !!this._dragContainsFiles(event);\n  }\n\n  apps = input.required<App[]>();\n  #appsEffect = effect(() => this.#shell.setAppBaseRoutes(this.apps()));\n  user?: YuvUser;\n  checkedForInitialRoute = false;\n  enableTenantSwitch = false;\n\n  config = input<ClientShellConfig>();\n  #shellConfigEffect = effect(\n    () => {\n      const cfg = this.config();\n      if (cfg) {\n        this.#shell.setShellConfig(cfg);\n      }\n    },\n    {\n      allowSignalWrites: true\n    }\n  );\n  shellConfig = this.#shell.shellConfig;\n\n  constructor() {\n    this.translate.onLangChange.subscribe(() => this._setCommands(true));\n    // this.router.events.pipe(\n    //   // filter(e => e instanceof NavigationStart)\n    // ).subscribe((e) => {\n    //   console.log(e);\n    // });\n\n    this.router.events\n      .pipe(\n        filter((e) => e instanceof NavigationEnd),\n        map((e) => e as NavigationEnd)\n      )\n      .subscribe((e: NavigationEnd) => this._processRouterNavigationEnd(e));\n\n    this.#device.init();\n\n    this.userService.user$.subscribe((user: YuvUser | undefined) => {\n      if (user) {\n        this.checkedForInitialRoute = !(!this.user || this.user.id !== user.id);\n        this.enableTenantSwitch = user.authorities.includes(UserRoles.MULTI_TENANT);\n      }\n      this.user = user;\n    });\n\n    window.addEventListener('storage', (evt) => {\n      if (evt.key === this.APP_LOGOUT_EVENT_KEY) {\n        this.appLogout(true);\n      }\n    });\n  }\n\n  openNotifications() {\n    this.router.navigate([{}]);\n  }\n\n  // getNotifications(id: string) {\n  //   return this.shellNotifications.getNotifications(id);\n  // }\n\n  private _dragContainsFiles(event: DragEvent): number {\n    // do not allow to drop files/images from iframes\n    if ((event.relatedTarget as Element)?.tagName.toLowerCase() === 'iframe') return 0;\n    return event.dataTransfer ? Array.from(event.dataTransfer.items || []).filter((i) => i.kind === 'file' && i.type).length : 0;\n  }\n\n  private _setCommands(update?: boolean) {\n    const commands: CommandPaletteCommand[] = this.apps().map((a: App, i: number) => ({\n      id: `nav.app.${i}`,\n      label: this.translate.instant('yuv.shell.cmd.app.open', { title: a.title as string }),\n      callback: () => this.router.navigate([a.path])\n    }));\n    commands.push({\n      id: `nav.shell.settings`,\n      label: this.translate.instant('yuv.shell.cmd.open.settings'),\n      callback: () => this.router.navigate(['settings'])\n    });\n    commands.push({\n      id: `nav.shell.logout`,\n      label: this.translate.instant('yuv.shell.cmd.logout'),\n      callback: () => this.appLogout()\n    });\n\n    if (update) this.commandPalette.updateCommands(commands);\n    else this.commandPalette.registerCommands(commands);\n  }\n\n  requestSubscription() {\n    if (!this.#swPush.isEnabled) {\n      console.log('Notification is not enabled.');\n      return;\n    }\n\n    this.#swPush.messages\n      .pipe(\n        tap((msg) => console.log({ msg })),\n        catchError((error) => {\n          console.log({ error });\n          return of(null);\n        })\n      )\n      .subscribe();\n  }\n\n  appLogout(triggeredFromOtherTab?: boolean) {\n    if (!triggeredFromOtherTab) {\n      // send storage event to logout all other open tabs\n      window.localStorage.setItem(this.APP_LOGOUT_EVENT_KEY, `${Date.now()}`);\n    }\n    this.userService.logout('');\n  }\n\n  private _processRouterNavigationEnd(e: NavigationEnd) {\n    if (!this.checkedForInitialRoute) {\n      this.checkedForInitialRoute = true;\n      // redirect to the page the user logged out from the last time\n      // but only if current route is not a deep link\n      const ignoreRoutes = ['', 'dashboard', 'index.html'].map((s) => `${Utils.getBaseHref()}${s}`.replace('//', '/'));\n      const currentRoute = this.routeWithBaseHref(this.router.routerState.snapshot.url);\n\n      if (this.userService.getCurrentUser() && !ignoreRoutes.includes(currentRoute)) {\n        // get persisted routes to decide where to redirect the logged in user to\n        this.auth\n          .getInitialRequestUri()\n          .pipe(switchMap((res) => this.auth.resetInitialRequestUri().pipe(map((_) => res))))\n          .subscribe((res: { uri: string; timestamp: number }) => {\n            const loginRes = res && !ignoreRoutes.includes(res.uri) ? res : null;\n            if (loginRes) this.router.navigateByUrl(loginRes.uri);\n          });\n      }\n    }\n  }\n\n  private routeWithBaseHref(r: string): string {\n    return `${Utils.getBaseHref()}${r}`.replace('//', '/');\n  }\n\n  ngOnInit(): void {\n    this.#shell._init();\n    clientShellRoutes.forEach((route: Route) => {\n      this.router.config.push(route);\n    });\n    this.router.resetConfig(this.router.config);\n    this._setCommands();\n    this.requestSubscription();\n  }\n}\n","<yuv-metadata-default-templates></yuv-metadata-default-templates>\n<!-- <yuv-focus-indicator /> -->\n\n<!-- gloabl busy indicator -->\n@if (busy$ | async) {\n  <div class=\"yuv-loader-linear\"></div>\n}\n<nav class=\"apps\" [inert]=\"asideOutlet.isActivated\" [attr.aria-label]=\"'yuv.shell.naviagtion.aria.label' | translate\">\n  <div class=\"app-icon\">\n    <button routerLink=\"/\" routerLinkActive=\"active\" [attr.aria-label]=\"'yuv.shell.logo.aria.label' | translate\">\n      <yvc-icon [svg]=\"shellConfig().icons!.appIcon!\"></yvc-icon>\n    </button>\n  </div>\n  <ul class=\"apps\">\n    @for (a of apps(); track a.path) {\n      <li>\n        <button routerLinkActive=\"active\" [routerLink]=\"a.path\" title=\"{{ a.title }}\">\n          <yvc-icon [svg]=\"a.icon\" title=\"{{ a.title }}\" *ngIf=\"a.icon\"></yvc-icon>\n          <!-- @if (getNotifications(a.id)) {\n            <div [ngClass]=\"'badge ' + getNotifications(a.id).maxLevel\">{{ getNotifications(a.id).count }}</div>\n          } -->\n        </button>\n      </li>\n    }\n  </ul>\n\n  <section class=\"actions\">\n    @if (showNotifications()) {\n      <button [routerLink]=\"[{ outlets: { aside: 'notifications' } }]\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.notifications.title' | translate }}\">\n        <yvc-icon [svg]=\"shellConfig().icons!.notifications!\"></yvc-icon>\n        @if (newNotifications(); as note) {\n          <div class=\"badge {{ note.maxLevel }}\">{{ note.count }}</div>\n        }\n      </button>\n    }\n    <button [routerLink]=\"['/settings']\" routerLinkActive=\"active\" title=\"{{ 'yuv.shell.settings.title' | translate }}\">\n      <yvc-icon [svg]=\"shellConfig().icons!.settings!\"></yvc-icon>\n    </button>\n    <button (click)=\"appLogout()\" title=\"{{ 'yuv.shell.cmd.logout' | translate }}\"><yvc-icon [svg]=\"shellConfig().icons!.logout!\"></yvc-icon></button>\n  </section>\n</nav>\n<main id=\"main-shell_content\" aria-label=\"main shell content\" [inert]=\"asideOutlet.isActivated\">\n  <router-outlet></router-outlet>\n</main>\n\n<!-- outlet for aside modals like notifications -->\n<div class=\"asideOutlet\" [hidden]=\"!asideOutlet.isActivated\">\n  <router-outlet name=\"aside\" #asideOutlet=\"outlet\"></router-outlet>\n</div>\n\n<div id=\"fi\" inert></div>\n"]}
@@ -0,0 +1,18 @@
1
+ import { Component, input } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { YvcIconModule } from '@yuuvis/components/icon';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@yuuvis/components/icon";
6
+ export class AppLogoComponent {
7
+ constructor() {
8
+ this.icon = input();
9
+ this.label = input.required();
10
+ }
11
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AppLogoComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
12
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: AppLogoComponent, isStandalone: true, selector: "yuv-app-logo", inputs: { icon: { classPropertyName: "icon", publicName: "icon", isSignal: true, isRequired: false, transformFunction: null }, label: { classPropertyName: "label", publicName: "label", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0, template: "@let ico = icon();\n@if(ico){\n<yvc-icon [svg]=\"ico\"></yvc-icon>\n}<span class=\"label\">{{label()}}</span>", styles: [":host{--band-color: var(--text-color-body);--band-height: var(--client-shell-bar-icon-size);--band-icon-size: 32px;--text-color: var(--main-background);--icon-color: var(--text-color);--icon-background-color: transparent;display:flex;align-items:center;padding:0 calc(var(--app-pane-padding) / 1) 0 calc(var(--app-pane-padding) / 2);height:calc(var(--client-shell-bar-icon-size));font-size:var(--font-caption);line-height:1em;cursor:pointer;background-color:var(--band-color);color:var(--text-color)}:host yvc-icon{--icon-size: var(--band-icon-size);grid-column:2;grid-row:1;border-radius:4px;margin-inline-end:calc(var(--app-pane-padding) / 2);opacity:.5;color:var(--icon-color);background-color:var(--icon-background-color)}:host .label{line-height:1em}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: YvcIconModule }, { kind: "component", type: i1.Icon, selector: "yvc-icon", inputs: ["label", "svg", "svgSrc"] }] }); }
13
+ }
14
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AppLogoComponent, decorators: [{
15
+ type: Component,
16
+ args: [{ selector: 'yuv-app-logo', standalone: true, imports: [CommonModule, YvcIconModule], template: "@let ico = icon();\n@if(ico){\n<yvc-icon [svg]=\"ico\"></yvc-icon>\n}<span class=\"label\">{{label()}}</span>", styles: [":host{--band-color: var(--text-color-body);--band-height: var(--client-shell-bar-icon-size);--band-icon-size: 32px;--text-color: var(--main-background);--icon-color: var(--text-color);--icon-background-color: transparent;display:flex;align-items:center;padding:0 calc(var(--app-pane-padding) / 1) 0 calc(var(--app-pane-padding) / 2);height:calc(var(--client-shell-bar-icon-size));font-size:var(--font-caption);line-height:1em;cursor:pointer;background-color:var(--band-color);color:var(--text-color)}:host yvc-icon{--icon-size: var(--band-icon-size);grid-column:2;grid-row:1;border-radius:4px;margin-inline-end:calc(var(--app-pane-padding) / 2);opacity:.5;color:var(--icon-color);background-color:var(--icon-background-color)}:host .label{line-height:1em}\n"] }]
17
+ }] });
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLWxvZ28uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy95dXV2aXMvY2xpZW50LXNoZWxsL3NyYy9saWIvY29tcG9uZW50cy9hcHAtbG9nby9hcHAtbG9nby5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3l1dXZpcy9jbGllbnQtc2hlbGwvc3JjL2xpYi9jb21wb25lbnRzL2FwcC1sb2dvL2FwcC1sb2dvLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0seUJBQXlCLENBQUM7OztBQVN4RCxNQUFNLE9BQU8sZ0JBQWdCO0lBUDdCO1FBUUUsU0FBSSxHQUFHLEtBQUssRUFBVSxDQUFDO1FBQ3ZCLFVBQUssR0FBRyxLQUFLLENBQUMsUUFBUSxFQUFVLENBQUM7S0FDbEM7K0dBSFksZ0JBQWdCO21HQUFoQixnQkFBZ0IsZ1VDWDdCLCtHQUd1Qyw4eUJESTNCLFlBQVksOEJBQUUsYUFBYTs7NEZBSTFCLGdCQUFnQjtrQkFQNUIsU0FBUzsrQkFDRSxjQUFjLGNBQ1osSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgaW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBZdmNJY29uTW9kdWxlIH0gZnJvbSAnQHl1dXZpcy9jb21wb25lbnRzL2ljb24nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd5dXYtYXBwLWxvZ28nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBZdmNJY29uTW9kdWxlXSxcbiAgdGVtcGxhdGVVcmw6ICcuL2FwcC1sb2dvLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmw6ICcuL2FwcC1sb2dvLmNvbXBvbmVudC5zY3NzJ1xufSlcbmV4cG9ydCBjbGFzcyBBcHBMb2dvQ29tcG9uZW50IHtcbiAgaWNvbiA9IGlucHV0PHN0cmluZz4oKTtcbiAgbGFiZWwgPSBpbnB1dC5yZXF1aXJlZDxzdHJpbmc+KCk7XG59XG4iLCJAbGV0IGljbyA9IGljb24oKTtcbkBpZihpY28pe1xuPHl2Yy1pY29uIFtzdmddPVwiaWNvXCI+PC95dmMtaWNvbj5cbn08c3BhbiBjbGFzcz1cImxhYmVsXCI+e3tsYWJlbCgpfX08L3NwYW4+Il19
@@ -0,0 +1,26 @@
1
+ import { Directive, ElementRef, effect, inject, input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class InertDirective {
4
+ constructor() {
5
+ this.elRef = inject(ElementRef);
6
+ this.inert = input(false);
7
+ effect(() => {
8
+ const el = this.elRef.nativeElement;
9
+ if (this.inert())
10
+ el.setAttribute('inert', 'true');
11
+ else
12
+ el.removeAttribute('inert');
13
+ });
14
+ }
15
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InertDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
16
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.13", type: InertDirective, isStandalone: true, selector: "[inert]", inputs: { inert: { classPropertyName: "inert", publicName: "inert", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0 }); }
17
+ }
18
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InertDirective, decorators: [{
19
+ type: Directive,
20
+ args: [{
21
+ // eslint-disable-next-line @angular-eslint/directive-selector
22
+ selector: '[inert]',
23
+ standalone: true
24
+ }]
25
+ }], ctorParameters: () => [] });
26
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5lcnQuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy95dXV2aXMvY2xpZW50LXNoZWxsL3NyYy9saWIvZGlyZWN0aXZlcy9pbmVydC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBTzdFLE1BQU0sT0FBTyxjQUFjO0lBS3pCO1FBSlEsVUFBSyxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUVuQyxVQUFLLEdBQUcsS0FBSyxDQUFVLEtBQUssQ0FBQyxDQUFDO1FBRzVCLE1BQU0sQ0FBQyxHQUFHLEVBQUU7WUFDVixNQUFNLEVBQUUsR0FBZ0IsSUFBSSxDQUFDLEtBQUssQ0FBQyxhQUE0QixDQUFDO1lBQ2hFLElBQUksSUFBSSxDQUFDLEtBQUssRUFBRTtnQkFBRSxFQUFFLENBQUMsWUFBWSxDQUFDLE9BQU8sRUFBRSxNQUFNLENBQUMsQ0FBQzs7Z0JBQzlDLEVBQUUsQ0FBQyxlQUFlLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDbkMsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDOytHQVhVLGNBQWM7bUdBQWQsY0FBYzs7NEZBQWQsY0FBYztrQkFMMUIsU0FBUzttQkFBQztvQkFDVCw4REFBOEQ7b0JBQzlELFFBQVEsRUFBRSxTQUFTO29CQUNuQixVQUFVLEVBQUUsSUFBSTtpQkFDakIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEVsZW1lbnRSZWYsIGVmZmVjdCwgaW5qZWN0LCBpbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5ARGlyZWN0aXZlKHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEBhbmd1bGFyLWVzbGludC9kaXJlY3RpdmUtc2VsZWN0b3JcbiAgc2VsZWN0b3I6ICdbaW5lcnRdJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBJbmVydERpcmVjdGl2ZSB7XG4gIHByaXZhdGUgZWxSZWYgPSBpbmplY3QoRWxlbWVudFJlZik7XG5cbiAgaW5lcnQgPSBpbnB1dDxib29sZWFuPihmYWxzZSk7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgZWZmZWN0KCgpID0+IHtcbiAgICAgIGNvbnN0IGVsOiBIVE1MRWxlbWVudCA9IHRoaXMuZWxSZWYubmF0aXZlRWxlbWVudCBhcyBIVE1MRWxlbWVudDtcbiAgICAgIGlmICh0aGlzLmluZXJ0KCkpIGVsLnNldEF0dHJpYnV0ZSgnaW5lcnQnLCAndHJ1ZScpO1xuICAgICAgZWxzZSBlbC5yZW1vdmVBdHRyaWJ1dGUoJ2luZXJ0Jyk7XG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,15 @@
1
+ import { DashboardPageComponent } from './pages/dashboard/dashboard.component';
2
+ import { SettingsPageComponent } from './pages/settings/settings.component';
3
+ import { NotificationsPageComponent } from './pages/notifications/notifications.component';
4
+ import { WebShareTargetPageComponent } from './pages/web-share-target/web-share-target.component';
5
+ export const clientShellRoutes = [
6
+ { path: 'dashboard', component: DashboardPageComponent },
7
+ { path: 'notifications', component: NotificationsPageComponent, outlet: 'aside' },
8
+ { path: 'settings', component: SettingsPageComponent },
9
+ { path: 'web-share-target', component: WebShareTargetPageComponent },
10
+ // default route
11
+ { path: '', redirectTo: '/dashboard', pathMatch: 'full' },
12
+ // redirecting route
13
+ { path: '**', redirectTo: '/' }
14
+ ];
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGliLnJvdXRlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL2xpYnMveXV1dmlzL2NsaWVudC1zaGVsbC9zcmMvbGliL2xpYi5yb3V0ZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDL0UsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDNUUsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sK0NBQStDLENBQUM7QUFDM0YsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0scURBQXFELENBQUM7QUFFbEcsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQVk7SUFDeEMsRUFBRSxJQUFJLEVBQUUsV0FBVyxFQUFFLFNBQVMsRUFBRSxzQkFBc0IsRUFBRTtJQUN4RCxFQUFFLElBQUksRUFBRSxlQUFlLEVBQUUsU0FBUyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sRUFBRSxPQUFPLEVBQUU7SUFDakYsRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLFNBQVMsRUFBRSxxQkFBcUIsRUFBRTtJQUN0RCxFQUFFLElBQUksRUFBRSxrQkFBa0IsRUFBRSxTQUFTLEVBQUUsMkJBQTJCLEVBQUU7SUFDcEUsZ0JBQWdCO0lBQ2hCLEVBQUUsSUFBSSxFQUFFLEVBQUUsRUFBRSxVQUFVLEVBQUUsWUFBWSxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUU7SUFDekQsb0JBQW9CO0lBQ3BCLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsR0FBRyxFQUFFO0NBQ2hDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSb3V0ZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBEYXNoYm9hcmRQYWdlQ29tcG9uZW50IH0gZnJvbSAnLi9wYWdlcy9kYXNoYm9hcmQvZGFzaGJvYXJkLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTZXR0aW5nc1BhZ2VDb21wb25lbnQgfSBmcm9tICcuL3BhZ2VzL3NldHRpbmdzL3NldHRpbmdzLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBOb3RpZmljYXRpb25zUGFnZUNvbXBvbmVudCB9IGZyb20gJy4vcGFnZXMvbm90aWZpY2F0aW9ucy9ub3RpZmljYXRpb25zLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBXZWJTaGFyZVRhcmdldFBhZ2VDb21wb25lbnQgfSBmcm9tICcuL3BhZ2VzL3dlYi1zaGFyZS10YXJnZXQvd2ViLXNoYXJlLXRhcmdldC5jb21wb25lbnQnO1xuXG5leHBvcnQgY29uc3QgY2xpZW50U2hlbGxSb3V0ZXM6IFJvdXRlW10gPSBbXG4gIHsgcGF0aDogJ2Rhc2hib2FyZCcsIGNvbXBvbmVudDogRGFzaGJvYXJkUGFnZUNvbXBvbmVudCB9LFxuICB7IHBhdGg6ICdub3RpZmljYXRpb25zJywgY29tcG9uZW50OiBOb3RpZmljYXRpb25zUGFnZUNvbXBvbmVudCwgb3V0bGV0OiAnYXNpZGUnIH0sXG4gIHsgcGF0aDogJ3NldHRpbmdzJywgY29tcG9uZW50OiBTZXR0aW5nc1BhZ2VDb21wb25lbnQgfSxcbiAgeyBwYXRoOiAnd2ViLXNoYXJlLXRhcmdldCcsIGNvbXBvbmVudDogV2ViU2hhcmVUYXJnZXRQYWdlQ29tcG9uZW50IH0sXG4gIC8vIGRlZmF1bHQgcm91dGVcbiAgeyBwYXRoOiAnJywgcmVkaXJlY3RUbzogJy9kYXNoYm9hcmQnLCBwYXRoTWF0Y2g6ICdmdWxsJyB9LFxuICAvLyByZWRpcmVjdGluZyByb3V0ZVxuICB7IHBhdGg6ICcqKicsIHJlZGlyZWN0VG86ICcvJyB9XG5dO1xuIl19
@@ -0,0 +1,11 @@
1
+ import { Component } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class DashboardPageComponent {
4
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DashboardPageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
5
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DashboardPageComponent, selector: "yuv-dashboard", ngImport: i0, template: "yuuvis Momentum\n", styles: [":host{display:grid;align-items:center;justify-content:center;height:100%;overflow:hidden;color:var(--text-color-hint);font-size:var(--font-display)}\n"] }); }
6
+ }
7
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DashboardPageComponent, decorators: [{
8
+ type: Component,
9
+ args: [{ selector: 'yuv-dashboard', template: "yuuvis Momentum\n", styles: [":host{display:grid;align-items:center;justify-content:center;height:100%;overflow:hidden;color:var(--text-color-hint);font-size:var(--font-display)}\n"] }]
10
+ }] });
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGFzaGJvYXJkLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMveXV1dmlzL2NsaWVudC1zaGVsbC9zcmMvbGliL3BhZ2VzL2Rhc2hib2FyZC9kYXNoYm9hcmQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy95dXV2aXMvY2xpZW50LXNoZWxsL3NyYy9saWIvcGFnZXMvZGFzaGJvYXJkL2Rhc2hib2FyZC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU8xQyxNQUFNLE9BQU8sc0JBQXNCOytHQUF0QixzQkFBc0I7bUdBQXRCLHNCQUFzQixxRENQbkMsbUJBQ0E7OzRGRE1hLHNCQUFzQjtrQkFMbEMsU0FBUzsrQkFDRSxlQUFlIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3l1di1kYXNoYm9hcmQnLFxuICB0ZW1wbGF0ZVVybDogJy4vZGFzaGJvYXJkLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vZGFzaGJvYXJkLmNvbXBvbmVudC5zY3NzJ11cbn0pXG5leHBvcnQgY2xhc3MgRGFzaGJvYXJkUGFnZUNvbXBvbmVudCB7fVxuIiwieXV1dmlzIE1vbWVudHVtXG4iXX0=