ngx-edu-sharing-metaqs2 0.9.32 → 0.9.35
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/README.md +31 -0
- package/_index.scss +8 -0
- package/esm2022/lib/collection-count-history/collection-count-history.component.mjs +46 -32
- package/esm2022/lib/collection-count-history/monthpicker/monthpicker.component.mjs +7 -4
- package/esm2022/lib/components/donut-chart/donut-chart.component.mjs +14 -14
- package/esm2022/lib/components/donut-chart/donut-chart.model.mjs +1 -1
- package/esm2022/lib/components/donut-chart/donut-chart.pipe.mjs +6 -5
- package/esm2022/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.mjs +79 -0
- package/esm2022/lib/components/filter/datepicker/datepicker.component.mjs +3 -8
- package/esm2022/lib/components/node-list/node-list.component.mjs +13 -9
- package/esm2022/lib/components/quality-matrix/quality_matrix.mjs +195 -36
- package/esm2022/lib/components/quality-matrix/scroll-marker.directive.mjs +17 -0
- package/esm2022/lib/config-helper.service.mjs +5 -4
- package/esm2022/lib/core/tooltip.service.mjs +146 -0
- package/esm2022/lib/counts-with-history/counts-with-history.component.mjs +87 -84
- package/esm2022/lib/java-api/api/authProxyController.service.mjs +12 -93
- package/esm2022/lib/java-api/api/collectionAPI.service.mjs +91 -178
- package/esm2022/lib/java-api/api/editorsAPI.service.mjs +14 -102
- package/esm2022/lib/java-api/api/filterAPI.service.mjs +50 -129
- package/esm2022/lib/java-api/api/replicationSourceAPI.service.mjs +20 -130
- package/esm2022/lib/java-api/api.base.service.mjs +66 -0
- package/esm2022/lib/java-api/configuration.mjs +9 -1
- package/esm2022/lib/java-api/model/missingAttributeResult.mjs +2 -0
- package/esm2022/lib/java-api/model/models.mjs +2 -1
- package/esm2022/lib/ng-meta-widgets-lib.module.mjs +19 -12
- package/esm2022/lib/tree-collection-details/tree-collection-details.component.mjs +3 -7
- package/esm2022/lib/tree-search-counts/tree-search-counts.component.mjs +4 -6
- package/esm2022/public-api.mjs +6 -4
- package/esm2022/web-components.mjs +36 -0
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs +1170 -1239
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs.map +1 -1
- package/lib/collection-count-history/collection-count-history.component.d.ts +1 -0
- package/lib/collection-count-history/monthpicker/monthpicker.component.d.ts +2 -1
- package/lib/components/donut-chart/donut-chart.component.d.ts +1 -1
- package/lib/components/donut-chart/donut-chart.model.d.ts +1 -0
- package/lib/components/donut-chart/donut-chart.pipe.d.ts +1 -1
- package/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.d.ts +14 -0
- package/lib/components/node-list/node-list.component.d.ts +9 -5
- package/lib/components/quality-matrix/quality_matrix.d.ts +25 -6
- package/lib/components/quality-matrix/scroll-marker.directive.d.ts +7 -0
- package/lib/config-helper.service.d.ts +2 -0
- package/lib/core/tooltip.service.d.ts +61 -0
- package/lib/counts-with-history/counts-with-history.component.d.ts +27 -25
- package/lib/java-api/api/authProxyController.service.d.ts +4 -9
- package/lib/java-api/api/collectionAPI.service.d.ts +47 -25
- package/lib/java-api/api/editorsAPI.service.d.ts +5 -10
- package/lib/java-api/api/filterAPI.service.d.ts +27 -9
- package/lib/java-api/api/replicationSourceAPI.service.d.ts +4 -9
- package/lib/java-api/api.base.service.d.ts +12 -0
- package/lib/java-api/configuration.d.ts +3 -1
- package/lib/java-api/model/missingAttributeResult.d.ts +16 -0
- package/lib/java-api/model/models.d.ts +1 -0
- package/lib/ng-meta-widgets-lib.module.d.ts +34 -32
- package/package.json +4 -1
- package/public-api.d.ts +5 -3
- package/web-components.d.ts +7 -0
- package/esm2022/lib/materialtypes-by-sources/materialtypes-by-sources.component.mjs +0 -148
- package/lib/materialtypes-by-sources/materialtypes-by-sources.component.d.ts +0 -43
|
@@ -1,7 +1,8 @@
|
|
|
1
|
-
import { Inject, Injectable } from '@angular/core';
|
|
1
|
+
import { Inject, Injectable, InjectionToken } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
export class NgMetaWidgetsLibConfiguration {
|
|
4
4
|
}
|
|
5
|
+
export const NG_META_WIDGETS_LIB_CONFIGURATION = new InjectionToken('NG_META_WIDGETS_LIB_CONFIGURATION');
|
|
5
6
|
/**
|
|
6
7
|
* helper class to provide configuration values
|
|
7
8
|
*/
|
|
@@ -19,13 +20,13 @@ export class ConfigHelperService {
|
|
|
19
20
|
get eduSharingPath() {
|
|
20
21
|
return this.config.eduSharingPath;
|
|
21
22
|
}
|
|
22
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConfigHelperService, deps: [{ token:
|
|
23
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConfigHelperService, deps: [{ token: NG_META_WIDGETS_LIB_CONFIGURATION }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
23
24
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConfigHelperService }); }
|
|
24
25
|
}
|
|
25
26
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConfigHelperService, decorators: [{
|
|
26
27
|
type: Injectable
|
|
27
28
|
}], ctorParameters: () => [{ type: NgMetaWidgetsLibConfiguration, decorators: [{
|
|
28
29
|
type: Inject,
|
|
29
|
-
args: [
|
|
30
|
+
args: [NG_META_WIDGETS_LIB_CONFIGURATION]
|
|
30
31
|
}] }] });
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLWhlbHBlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctbWV0YS13aWRnZXRzLWxpYi9zcmMvbGliL2NvbmZpZy1oZWxwZXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLFVBQVUsRUFBRSxjQUFjLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRW5FLE1BQU0sT0FBTyw2QkFBNkI7Q0FHekM7QUFFRCxNQUFNLENBQUMsTUFBTSxpQ0FBaUMsR0FBRyxJQUFJLGNBQWMsQ0FDakUsbUNBQW1DLENBQ3BDLENBQUM7QUFFRjs7R0FFRztBQUVILE1BQU0sT0FBTyxtQkFBbUI7SUFDOUIsWUFBOEQsTUFBcUM7UUFBckMsV0FBTSxHQUFOLE1BQU0sQ0FBK0I7UUFDakcscUNBQXFDO0lBQ3ZDLENBQUM7SUFFTSxXQUFXLENBQUMsTUFBOEM7UUFDL0QsSUFBSSxDQUFDLE1BQU0sR0FBRyxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLE1BQU0sRUFBRSxDQUFDO0lBQzlDLENBQUM7SUFFRCxJQUFXLE9BQU87UUFDaEIsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQztJQUM3QixDQUFDO0lBRUQsSUFBVyxjQUFjO1FBQ3ZCLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxjQUFjLENBQUM7SUFDcEMsQ0FBQzsrR0FmVSxtQkFBbUIsa0JBQ1YsaUNBQWlDO21IQUQxQyxtQkFBbUI7OzRGQUFuQixtQkFBbUI7a0JBRC9CLFVBQVU7OzBCQUVJLE1BQU07MkJBQUMsaUNBQWlDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0LCBJbmplY3RhYmxlLCBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgY2xhc3MgTmdNZXRhV2lkZ2V0c0xpYkNvbmZpZ3VyYXRpb24ge1xuICBhcGlQYXRoOiBzdHJpbmc7XG4gIGVkdVNoYXJpbmdQYXRoOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBjb25zdCBOR19NRVRBX1dJREdFVFNfTElCX0NPTkZJR1VSQVRJT04gPSBuZXcgSW5qZWN0aW9uVG9rZW48TmdNZXRhV2lkZ2V0c0xpYkNvbmZpZ3VyYXRpb24+KFxuICAnTkdfTUVUQV9XSURHRVRTX0xJQl9DT05GSUdVUkFUSU9OJ1xuKTtcblxuLyoqXG4gKiBoZWxwZXIgY2xhc3MgdG8gcHJvdmlkZSBjb25maWd1cmF0aW9uIHZhbHVlc1xuICovXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgQ29uZmlnSGVscGVyU2VydmljZSBpbXBsZW1lbnRzIENvbmZpZ0hlbHBlciB7XG4gIGNvbnN0cnVjdG9yKEBJbmplY3QoTkdfTUVUQV9XSURHRVRTX0xJQl9DT05GSUdVUkFUSU9OKSBwdWJsaWMgY29uZmlnOiBOZ01ldGFXaWRnZXRzTGliQ29uZmlndXJhdGlvbikge1xuICAgIC8vIGFwcGx5IGFwaSBwYXRoIGdpdmVuIGluIHdpbmRvdyB1cmxcbiAgfVxuXG4gIHB1YmxpYyBwYXRjaENvbmZpZyhjb25maWc6IFBhcnRpYWw8TmdNZXRhV2lkZ2V0c0xpYkNvbmZpZ3VyYXRpb24+KSB7XG4gICAgdGhpcy5jb25maWcgPSB7IC4uLnRoaXMuY29uZmlnLCAuLi5jb25maWcgfTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgYXBpUGF0aCgpIHtcbiAgICByZXR1cm4gdGhpcy5jb25maWcuYXBpUGF0aDtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgZWR1U2hhcmluZ1BhdGgoKSB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLmVkdVNoYXJpbmdQYXRoO1xuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29uZmlnSGVscGVyIHtcbiAgZ2V0IGFwaVBhdGgoKTogc3RyaW5nO1xuXG4gIGdldCBlZHVTaGFyaW5nUGF0aCgpOiBzdHJpbmc7XG5cbiAgcGF0Y2hDb25maWcoY29uZmlnOiBQYXJ0aWFsPE5nTWV0YVdpZGdldHNMaWJDb25maWd1cmF0aW9uPik6IHZvaWQ7XG59XG4iXX0=
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
import { inject, Injectable, InjectionToken, Injector, Renderer2 } from '@angular/core';
|
|
2
|
+
import { Overlay, OverlayRef } from '@angular/cdk/overlay';
|
|
3
|
+
import { ComponentPortal } from '@angular/cdk/portal';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export const TOOLTIP_DATA = new InjectionToken('TOOLTIP_DATA');
|
|
6
|
+
export const TOOLTIP_REF = new InjectionToken('TOOLTIP_REF');
|
|
7
|
+
const noop = Object.freeze(() => { });
|
|
8
|
+
/**
|
|
9
|
+
* @internal
|
|
10
|
+
*/
|
|
11
|
+
export class TooltipRefImpl {
|
|
12
|
+
constructor(renderer, overlay, openTimeout = 0, closeTimeout = 0, onOpen = noop, onClose = noop) {
|
|
13
|
+
this.renderer = renderer;
|
|
14
|
+
this.overlay = overlay;
|
|
15
|
+
this.openTimeout = openTimeout;
|
|
16
|
+
this.closeTimeout = closeTimeout;
|
|
17
|
+
this.onOpen = onOpen;
|
|
18
|
+
this.onClose = onClose;
|
|
19
|
+
}
|
|
20
|
+
cancelCloseAction() {
|
|
21
|
+
clearTimeout(this.closeTimeoutRef);
|
|
22
|
+
this.closeTimeoutRef = undefined;
|
|
23
|
+
}
|
|
24
|
+
open() {
|
|
25
|
+
this.openTimeoutRef = setTimeout(this.openTooltip.bind(this), this.openTimeout); // typescript bug (typescript mistakenly uses NodeJS.Timeout instead of number as return type of setTimeout)
|
|
26
|
+
}
|
|
27
|
+
openTooltip() {
|
|
28
|
+
this.openTimeoutRef = undefined;
|
|
29
|
+
this.component = this.overlay.attach(this.portal);
|
|
30
|
+
this.unlistenMouseEnter = this.renderer.listen(this.overlay.overlayElement, 'mouseenter', this.cancelCloseAction.bind(this));
|
|
31
|
+
this.unlistenMouseLeave = this.renderer.listen(this.overlay.overlayElement, 'mouseleave', this.markForClose.bind(this));
|
|
32
|
+
this.onOpen();
|
|
33
|
+
}
|
|
34
|
+
markForClose() {
|
|
35
|
+
if (this.openTimeoutRef) {
|
|
36
|
+
// not yet opened, cancel open action
|
|
37
|
+
clearTimeout(this.openTimeoutRef);
|
|
38
|
+
this.openTimeoutRef = undefined;
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
if (this.closeTimeoutRef) {
|
|
42
|
+
// already marked for close
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
this.closeTimeoutRef = setTimeout(this.closeTooltip.bind(this), this.closeTimeout); // typescript bug (typescript mistakenly uses NodeJS.Timeout instead of number as return type of setTimeout)
|
|
46
|
+
}
|
|
47
|
+
closeTooltip() {
|
|
48
|
+
this.overlay.detach();
|
|
49
|
+
this.onClose();
|
|
50
|
+
this.closeTimeoutRef = undefined;
|
|
51
|
+
this.unlistenMouseEnter?.();
|
|
52
|
+
this.unlistenMouseLeave?.();
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
export const centerPositionStrategy = [
|
|
56
|
+
{
|
|
57
|
+
originX: 'center',
|
|
58
|
+
originY: 'center',
|
|
59
|
+
overlayX: 'start',
|
|
60
|
+
overlayY: 'top',
|
|
61
|
+
},
|
|
62
|
+
{
|
|
63
|
+
originX: 'center',
|
|
64
|
+
originY: 'center',
|
|
65
|
+
overlayX: 'start',
|
|
66
|
+
overlayY: 'bottom',
|
|
67
|
+
},
|
|
68
|
+
{
|
|
69
|
+
originX: 'center',
|
|
70
|
+
originY: 'center',
|
|
71
|
+
overlayX: 'end',
|
|
72
|
+
overlayY: 'top',
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
originX: 'center',
|
|
76
|
+
originY: 'center',
|
|
77
|
+
overlayX: 'end',
|
|
78
|
+
overlayY: 'bottom',
|
|
79
|
+
},
|
|
80
|
+
];
|
|
81
|
+
export const borderPositionStrategy = [
|
|
82
|
+
{
|
|
83
|
+
originX: 'center',
|
|
84
|
+
originY: 'bottom',
|
|
85
|
+
overlayX: 'start',
|
|
86
|
+
overlayY: 'top',
|
|
87
|
+
},
|
|
88
|
+
{
|
|
89
|
+
originX: 'center',
|
|
90
|
+
originY: 'bottom',
|
|
91
|
+
overlayX: 'end',
|
|
92
|
+
overlayY: 'top',
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
originX: 'center',
|
|
96
|
+
originY: 'top',
|
|
97
|
+
overlayX: 'start',
|
|
98
|
+
overlayY: 'bottom',
|
|
99
|
+
},
|
|
100
|
+
{
|
|
101
|
+
originX: 'center',
|
|
102
|
+
originY: 'top',
|
|
103
|
+
overlayX: 'end',
|
|
104
|
+
overlayY: 'bottom',
|
|
105
|
+
},
|
|
106
|
+
];
|
|
107
|
+
export class TooltipService {
|
|
108
|
+
constructor() {
|
|
109
|
+
this.overlay = inject(Overlay);
|
|
110
|
+
this.renderer = inject(Renderer2);
|
|
111
|
+
}
|
|
112
|
+
create(component, config) {
|
|
113
|
+
const overlayRef = this.overlay.create(config.overlayConfig ?? {
|
|
114
|
+
hasBackdrop: false,
|
|
115
|
+
disposeOnNavigation: true,
|
|
116
|
+
});
|
|
117
|
+
const tooltipRef = new TooltipRefImpl(this.renderer, overlayRef, config.openTimeout ?? 0, config.closeTimeout ?? 0, config.onOpen ?? noop, config.onClose ?? noop);
|
|
118
|
+
tooltipRef.portal = new ComponentPortal(component, null, Injector.create({
|
|
119
|
+
providers: [
|
|
120
|
+
{
|
|
121
|
+
provide: OverlayRef,
|
|
122
|
+
useValue: overlayRef,
|
|
123
|
+
},
|
|
124
|
+
{
|
|
125
|
+
provide: TOOLTIP_DATA,
|
|
126
|
+
useValue: config.data,
|
|
127
|
+
},
|
|
128
|
+
{
|
|
129
|
+
provide: TOOLTIP_REF,
|
|
130
|
+
useValue: tooltipRef,
|
|
131
|
+
},
|
|
132
|
+
],
|
|
133
|
+
parent: config.injector,
|
|
134
|
+
}));
|
|
135
|
+
return tooltipRef;
|
|
136
|
+
}
|
|
137
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TooltipService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
138
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TooltipService, providedIn: 'root' }); }
|
|
139
|
+
}
|
|
140
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TooltipService, decorators: [{
|
|
141
|
+
type: Injectable,
|
|
142
|
+
args: [{
|
|
143
|
+
providedIn: 'root',
|
|
144
|
+
}]
|
|
145
|
+
}] });
|
|
146
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.service.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/core/tooltip.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentRef, inject, Injectable, InjectionToken, Injector, Renderer2, Type } from '@angular/core';\nimport { ConnectedPosition, Overlay, OverlayConfig, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\n\nexport const TOOLTIP_DATA = new InjectionToken<unknown>('TOOLTIP_DATA');\nexport const TOOLTIP_REF = new InjectionToken<TooltipRef<unknown>>('TOOLTIP_REF');\n\nexport interface TooltipRef<T> {\n  markForClose(): void;\n  cancelCloseAction(): void;\n\n  readonly overlay: OverlayRef;\n  readonly component: ComponentRef<T> | undefined;\n\n  open(): void;\n}\n\nconst noop = Object.freeze(() => {});\n\n/**\n * @internal\n */\nexport class TooltipRefImpl<T> implements TooltipRef<T> {\n  public openTimeoutRef: number | undefined;\n  public closeTimeoutRef: number | undefined;\n  public component: ComponentRef<T>;\n  private unlistenMouseEnter: () => void | undefined;\n  private unlistenMouseLeave: () => void | undefined;\n  public portal: ComponentPortal<T>;\n\n  public constructor(\n    private readonly renderer: Renderer2,\n    public readonly overlay: OverlayRef,\n    private readonly openTimeout = 0,\n    private readonly closeTimeout = 0,\n    private readonly onOpen = noop,\n    private readonly onClose = noop\n  ) {}\n\n  public cancelCloseAction(): void {\n    clearTimeout(this.closeTimeoutRef);\n    this.closeTimeoutRef = undefined;\n  }\n\n  public open() {\n    this.openTimeoutRef = setTimeout(this.openTooltip.bind(this), this.openTimeout) as unknown as number; // typescript bug (typescript mistakenly uses NodeJS.Timeout instead of number as return type of setTimeout)\n  }\n\n  public openTooltip() {\n    this.openTimeoutRef = undefined;\n    this.component = this.overlay.attach(this.portal);\n    this.unlistenMouseEnter = this.renderer.listen(\n      this.overlay.overlayElement,\n      'mouseenter',\n      this.cancelCloseAction.bind(this)\n    );\n    this.unlistenMouseLeave = this.renderer.listen(\n      this.overlay.overlayElement,\n      'mouseleave',\n      this.markForClose.bind(this)\n    );\n    this.onOpen();\n  }\n\n  public markForClose(): void {\n    if (this.openTimeoutRef) {\n      // not yet opened, cancel open action\n      clearTimeout(this.openTimeoutRef);\n      this.openTimeoutRef = undefined;\n      return;\n    }\n    if (this.closeTimeoutRef) {\n      // already marked for close\n      return;\n    }\n    this.closeTimeoutRef = setTimeout(this.closeTooltip.bind(this), this.closeTimeout) as unknown as number; // typescript bug (typescript mistakenly uses NodeJS.Timeout instead of number as return type of setTimeout)\n  }\n\n  public closeTooltip() {\n    this.overlay.detach();\n    this.onClose();\n    this.closeTimeoutRef = undefined;\n    this.unlistenMouseEnter?.();\n    this.unlistenMouseLeave?.();\n  }\n}\n\nexport const centerPositionStrategy: Array<ConnectedPosition> = [\n  {\n    originX: 'center',\n    originY: 'center',\n    overlayX: 'start',\n    overlayY: 'top',\n  },\n  {\n    originX: 'center',\n    originY: 'center',\n    overlayX: 'start',\n    overlayY: 'bottom',\n  },\n  {\n    originX: 'center',\n    originY: 'center',\n    overlayX: 'end',\n    overlayY: 'top',\n  },\n  {\n    originX: 'center',\n    originY: 'center',\n    overlayX: 'end',\n    overlayY: 'bottom',\n  },\n];\n\nexport const borderPositionStrategy: Array<ConnectedPosition> = [\n  {\n    originX: 'center',\n    originY: 'bottom',\n    overlayX: 'start',\n    overlayY: 'top',\n  },\n  {\n    originX: 'center',\n    originY: 'bottom',\n    overlayX: 'end',\n    overlayY: 'top',\n  },\n  {\n    originX: 'center',\n    originY: 'top',\n    overlayX: 'start',\n    overlayY: 'bottom',\n  },\n  {\n    originX: 'center',\n    originY: 'top',\n    overlayX: 'end',\n    overlayY: 'bottom',\n  },\n];\n\n// generic argument T is necessary for typescript to check the data passed to the config\nexport type DataTooltipHelper<T> = {\n  // this is a workaround to make typescript differentiate between DataTooltipHelper and unknown\n  [key in never]: T;\n};\n\nexport interface DataTooltip<T> extends DataTooltipHelper<T> {}\n\ntype DataTypeOfTooltip<T> = T extends DataTooltip<infer U> ? U : never;\n\nexport interface TooltipConfig<TData> {\n  injector?: Injector;\n  closeTimeout?: number;\n  openTimeout?: number;\n  data: TData;\n  onClose?: () => void;\n  onOpen?: () => void;\n  overlayConfig?: OverlayConfig;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class TooltipService {\n  private readonly overlay = inject(Overlay);\n  private readonly renderer = inject(Renderer2);\n\n  public create<TComponent extends DataTooltip<DataTypeOfTooltip<TComponent>>>(\n    component: Type<TComponent>,\n    config: TooltipConfig<DataTypeOfTooltip<TComponent>>\n  ): TooltipRef<TComponent> {\n    const overlayRef = this.overlay.create(\n      config.overlayConfig ?? {\n        hasBackdrop: false,\n        disposeOnNavigation: true,\n      }\n    );\n\n    const tooltipRef = new TooltipRefImpl<TComponent>(\n      this.renderer,\n      overlayRef,\n      config.openTimeout ?? 0,\n      config.closeTimeout ?? 0,\n      config.onOpen ?? noop,\n      config.onClose ?? noop\n    );\n    tooltipRef.portal = new ComponentPortal(\n      component,\n      null,\n      Injector.create({\n        providers: [\n          {\n            provide: OverlayRef,\n            useValue: overlayRef,\n          },\n          {\n            provide: TOOLTIP_DATA,\n            useValue: config.data,\n          },\n          {\n            provide: TOOLTIP_REF,\n            useValue: tooltipRef,\n          },\n        ],\n        parent: config.injector,\n      })\n    );\n\n    return tooltipRef;\n  }\n}\n"]}
|
|
@@ -1,15 +1,17 @@
|
|
|
1
|
-
import { Component, Input, signal } from '@angular/core';
|
|
1
|
+
import { Component, computed, effect, Input, signal } from '@angular/core';
|
|
2
2
|
import { MatCard, MatCardContent, MatCardHeader, MatCardModule, MatCardTitle } from '@angular/material/card';
|
|
3
3
|
import { MatTooltip } from '@angular/material/tooltip';
|
|
4
4
|
import { MatHeaderRow, MatHeaderRowDef, MatRow, MatRowDef, MatTable, MatTableModule } from '@angular/material/table';
|
|
5
|
-
import { BehaviorSubject,
|
|
5
|
+
import { BehaviorSubject, zip } from 'rxjs';
|
|
6
6
|
import { TranslateModule } from '@ngx-translate/core';
|
|
7
|
-
import { AsyncPipe,
|
|
7
|
+
import { AsyncPipe, NgForOf, NgIf } from '@angular/common';
|
|
8
8
|
import { DateTime } from 'luxon';
|
|
9
|
-
import { finalize, map, skipWhile,
|
|
10
|
-
import { FormControl, FormGroup } from '@angular/forms';
|
|
9
|
+
import { filter, finalize, map, skipWhile, switchMap, take, tap } from 'rxjs/operators';
|
|
10
|
+
import { FormControl, FormGroup, FormsModule } from '@angular/forms';
|
|
11
11
|
import { DatepickerComponent } from '../components/filter/datepicker/datepicker.component';
|
|
12
12
|
import { MatIcon } from '@angular/material/icon';
|
|
13
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
14
|
+
import { MatSlideToggle } from '@angular/material/slide-toggle';
|
|
13
15
|
import { ProgressSpinnerComponent } from '../components/loading_indicator/progress-spinner/progress-spinner.component';
|
|
14
16
|
import * as i0 from "@angular/core";
|
|
15
17
|
import * as i1 from "../meta-api.service";
|
|
@@ -17,16 +19,19 @@ import * as i2 from "../components/editorial-link-service/editorial-link.service
|
|
|
17
19
|
import * as i3 from "@angular/material/card";
|
|
18
20
|
import * as i4 from "@angular/material/table";
|
|
19
21
|
import * as i5 from "@ngx-translate/core";
|
|
22
|
+
import * as i6 from "@angular/forms";
|
|
20
23
|
export class CountsWithHistoryComponent {
|
|
21
|
-
constructor(metaApi, linkService) {
|
|
24
|
+
constructor(metaApi, destroyRef, linkService) {
|
|
22
25
|
this.metaApi = metaApi;
|
|
26
|
+
this.destroyRef = destroyRef;
|
|
23
27
|
this.linkService = linkService;
|
|
24
|
-
this.
|
|
28
|
+
this.loadingCount = signal(0);
|
|
29
|
+
this.isLoading = computed(() => this.loadingCount() > 0);
|
|
25
30
|
this.timeFilterLoaded = signal(false);
|
|
26
|
-
this.destroyed$ = new Subject();
|
|
27
31
|
this.recentTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
28
32
|
this.pastTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
29
|
-
this.
|
|
33
|
+
this.DateTime = DateTime;
|
|
34
|
+
this.columns = signal([]);
|
|
30
35
|
this.apiMethod = 'getMaterialTypeCountsByReplicationSource';
|
|
31
36
|
this.columnTranslationkey = null;
|
|
32
37
|
/* In this widget's backend we do have data for today
|
|
@@ -36,72 +41,83 @@ export class CountsWithHistoryComponent {
|
|
|
36
41
|
start: new FormControl(),
|
|
37
42
|
end: new FormControl(DateTime.utc().startOf('day'), { nonNullable: true }),
|
|
38
43
|
});
|
|
39
|
-
this.
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
return this.columns.map((c) => c.id + '_recent');
|
|
49
|
-
}
|
|
50
|
-
get pastColumns() {
|
|
51
|
-
return this.columns.map((c) => c.id + '_past');
|
|
52
|
-
}
|
|
53
|
-
get allColumns() {
|
|
54
|
-
return this.pastColumns.flatMap((e, i) => [e, this.recentColumns[i]]);
|
|
55
|
-
}
|
|
56
|
-
registerDateRangeFilter() {
|
|
57
|
-
/* we cannot use the takeUntilDestroyed() helper here,
|
|
58
|
-
* because this method is indirectly called in the finalization of the getTimerangeFilter() call
|
|
59
|
-
*/
|
|
60
|
-
this.range.controls.end.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((date) => {
|
|
61
|
-
this.getCountByDate(date).subscribe((response) => {
|
|
62
|
-
this.recentTypeCount$.next(response);
|
|
63
|
-
});
|
|
44
|
+
this.isHistoryEnabled = signal(false);
|
|
45
|
+
this.allColumns = computed(() => {
|
|
46
|
+
if (!this.isHistoryEnabled()) {
|
|
47
|
+
return this.recentColumns();
|
|
48
|
+
}
|
|
49
|
+
return this.pastColumns().flatMap((e, i) => [e, this.recentColumns()[i]]);
|
|
50
|
+
});
|
|
51
|
+
this.typeColumns = computed(() => {
|
|
52
|
+
return this.columns().map((c) => c.id + '_type');
|
|
64
53
|
});
|
|
65
|
-
this.
|
|
66
|
-
this.
|
|
67
|
-
|
|
68
|
-
|
|
54
|
+
this.recentColumns = computed(() => {
|
|
55
|
+
return this.columns().map((c) => c.id + '_recent');
|
|
56
|
+
});
|
|
57
|
+
this.pastColumns = computed(() => {
|
|
58
|
+
return this.columns().map((c) => c.id + '_past');
|
|
59
|
+
});
|
|
60
|
+
effect(() => {
|
|
61
|
+
this.range.controls.end.reset();
|
|
62
|
+
if (!this.isHistoryEnabled()) {
|
|
63
|
+
this.range.controls.start.setValue(this.range.controls.end.value.startOf('day'));
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
this.range.controls.start.reset();
|
|
67
|
+
}
|
|
69
68
|
});
|
|
70
69
|
}
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
.
|
|
74
|
-
.
|
|
70
|
+
ngOnInit() {
|
|
71
|
+
this.getAvailableDateRange()
|
|
72
|
+
.pipe(tap((rangeFilter) => {
|
|
73
|
+
const startDate = this.getStartDateOfRange(rangeFilter);
|
|
74
|
+
//this is to have a default value for the start date => the min date of the range
|
|
75
|
+
this.range.setControl('start', new FormControl(startDate, { nonNullable: true }));
|
|
76
|
+
}), finalize(() => {
|
|
75
77
|
this.timeFilterLoaded.set(true);
|
|
76
|
-
this.registerDateRangeFilter();
|
|
77
78
|
this.range.controls.start.reset();
|
|
78
|
-
}))
|
|
79
|
-
.subscribe(
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
79
|
+
}), takeUntilDestroyed(this.destroyRef))
|
|
80
|
+
.subscribe();
|
|
81
|
+
this.getCountByDate(this.range.controls.end.value)
|
|
82
|
+
.pipe(tap((response) => {
|
|
83
|
+
this.columns.set(response.columns.slice());
|
|
84
|
+
this.recentTypeCount$.next(response);
|
|
85
|
+
}), takeUntilDestroyed(this.destroyRef))
|
|
86
|
+
.subscribe();
|
|
87
|
+
this.getValuesInDateRange()
|
|
88
|
+
.pipe(tap(([past, recent]) => {
|
|
89
|
+
this.pastTypeCount$.next(past);
|
|
90
|
+
this.recentTypeCount$.next(recent);
|
|
91
|
+
}), takeUntilDestroyed(this.destroyRef))
|
|
92
|
+
.subscribe();
|
|
93
|
+
}
|
|
94
|
+
getStartDateOfRange(rangeFilter) {
|
|
95
|
+
return DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label, {
|
|
96
|
+
zone: 'utc',
|
|
97
|
+
}).startOf('day');
|
|
98
|
+
}
|
|
99
|
+
columnIdent(_index, col) {
|
|
100
|
+
return col.id;
|
|
101
|
+
}
|
|
102
|
+
getAvailableDateRange() {
|
|
103
|
+
this.loadingCount.update((it) => it + 1);
|
|
104
|
+
return this.metaApi.getTimerangeFilter().pipe(filter((filter) => filter != null), finalize(() => this.loadingCount.update((it) => it - 1)));
|
|
105
|
+
}
|
|
106
|
+
getValuesInDateRange() {
|
|
107
|
+
return this.range.valueChanges.pipe(filter((range) => !!range.start?.isValid && !!range.end?.isValid), switchMap((range) => {
|
|
108
|
+
return zip(this.getCountByDate(range.start), this.getCountByDate(range.end));
|
|
109
|
+
}));
|
|
88
110
|
}
|
|
89
111
|
getCountByDate(date) {
|
|
90
|
-
this.
|
|
112
|
+
this.loadingCount.update((it) => it + 1);
|
|
91
113
|
const filter = {
|
|
92
114
|
field: 'asOf',
|
|
93
115
|
values: [{ id: date.toISO({ includeOffset: false }), label: '' }],
|
|
94
116
|
};
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
// typesript gets confused here
|
|
100
|
-
// @ts-ignore
|
|
101
|
-
finalize(() => this.isLoading.set(false)), map((response) => {
|
|
102
|
-
response.rows.sort((a, b) => a.meta.label.localeCompare(b.meta.label));
|
|
103
|
-
return response;
|
|
104
|
-
}));
|
|
117
|
+
return this.metaApi[this.apiMethod]([filter]).pipe(finalize(() => this.loadingCount.update((it) => it - 1)), map((response) => ({
|
|
118
|
+
...response,
|
|
119
|
+
rows: response.rows.toSorted((a, b) => a.meta.label.localeCompare(b.meta.label)),
|
|
120
|
+
})));
|
|
105
121
|
}
|
|
106
122
|
pastTypeCount(row, columnid) {
|
|
107
123
|
if (!this.pastTypeCount$.value.rows.length) {
|
|
@@ -127,22 +143,8 @@ export class CountsWithHistoryComponent {
|
|
|
127
143
|
return this.linkService.createLinkForCountsWithHistory(this.sourceType, sourceId, issueId);
|
|
128
144
|
}));
|
|
129
145
|
}
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
this.getCountByDate(this.range.controls.end.value)
|
|
133
|
-
.pipe(tap((response) => {
|
|
134
|
-
this.columns = response.columns.slice();
|
|
135
|
-
}))
|
|
136
|
-
.subscribe((response) => {
|
|
137
|
-
this.recentTypeCount$.next(response);
|
|
138
|
-
});
|
|
139
|
-
}
|
|
140
|
-
ngOnDestroy() {
|
|
141
|
-
this.destroyed$.next();
|
|
142
|
-
this.destroyed$.complete();
|
|
143
|
-
}
|
|
144
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, deps: [{ token: i1.MetaApiService }, { token: i2.EditorialLinkService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
145
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CountsWithHistoryComponent, isStandalone: true, selector: "metaqs2-counts-with-history", inputs: { apiMethod: "apiMethod", columnTranslationkey: "columnTranslationkey", pageTitle: "pageTitle", sourceType: "sourceType" }, ngImport: i0, template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title data-test-id=\"page-title\">\n Qualit\u00E4tsmetrik: {{ pageTitle | translate}}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}\n </mat-card-title>\n </mat-card-header>\n <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <!-- show the filter after the values are loaded to avoid loading current data twice -->\n <mat-card-content *ngIf=\"timeFilterLoaded()\">\n <metaqs2-datepicker [disabled]=\"isLoading()\" [inputGroup]=\"range\" ></metaqs2-datepicker>\n </mat-card-content>\n</mat-card>\n<mat-card>\n <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"recentTypeCount$.value.rows\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"label-col\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Quelle</th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"label-col\"\n >\n {{row.meta.label}}\n </td>\n </ng-container>\n <!-- one column for each type -->\n <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n <th colspan=\"2\" mat-header-cell *matHeaderCellDef matTooltip=\"{{col.label}}\" >\n {{columnTranslationkey ? (columnTranslationkey + col.label | translate ) : col.label }}\n </th>\n </ng-container>\n <!-- one column for each type for the most current date-->\n <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_recent'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\" >{{ range.controls.end.value.toLocaleString(DateTime.DATE_SHORT) }}</th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\">\n <a [attr.href]=\"openInEditor(row.meta.id, col.id) | async\" target=\"editor_frontend\">{{ row.counts[col.id] ?? 'n/a' }}</a>\n </td>\n </ng-container>\n <!-- one column for each type for the older date-->\n <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_past'\">\n <th class=\"past-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\" >\n {{ pastTypeCount$.value.rows.length ? range.controls.start.value.toLocaleString(DateTime.DATE_SHORT) : 'no past data' }}\n </th>\n <td class=\"past-data-cell\" mat-cell *matCellDef=\"let row;\">\n <ng-container *ngIf=\"pastTypeCount$.value.rows.length && pastTypeCount(row, col.id) as trend\">\n <span [ngClass]=\"trend.trend\"> {{ trend.value ?? 'n/a' }}<mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" /></span>\n <span class=\"cdk-visually-hidden\">{{trend.trend}}</span>\n </ng-container>\n </td>\n </ng-container>\n <!-- generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat(typeColumns); sticky:true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"allColumns; sticky: true;\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(allColumns)\"></tr>\n\n </table>\n</mat-card>", styles: [".while-loading{filter:blur(2px)}tr:nth-child(2n){background-color:#e4e4e4}tr:nth-child(2n)>td.label-col,tr:nth-child(2n) td.recent-data-cell{border-right:1px solid white}tr:nth-child(odd){background-color:#fff}tr:nth-child(odd)>td.label-col,tr:nth-child(odd) td.recent-data-cell{border-right:1px solid #e4e4e4}td.label-col{text-align:left}.mat-mdc-header-cell,.mat-mdc-cell{text-align:center}.mat-mdc-header-cell a[href],.mat-mdc-cell a[href]{color:var(--mat-table-row-item-label-text-color);cursor:pointer;text-decoration:underline}.mat-mdc-header-cell a[href]:hover,.mat-mdc-cell a[href]:hover{text-decoration:underline}.mat-mdc-header-cell mat-icon,.mat-mdc-cell mat-icon{margin-left:5px;vertical-align:middle}.trending_down{color:#4abeff}.trending_up{color:#c20808}\n"], dependencies: [{ kind: "ngmodule", type: MatCardModule }, { kind: "component", type: i3.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "directive", type: i3.MatCardContent, selector: "mat-card-content" }, { kind: "component", type: i3.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i3.MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "directive", type: MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatTableModule }, { kind: "component", type: i4.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i4.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i4.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i4.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i4.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i4.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i4.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i4.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i4.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i4.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: DatepickerComponent, selector: "metaqs2-datepicker", inputs: ["disabled", "inputGroup"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: ProgressSpinnerComponent, selector: "metaqs2-progress-spinner", inputs: ["color", "diameter", "strokeWidth", "backdropEnabled", "positionGloballyCenter", "displayProgressSpinner"] }] }); }
|
|
146
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, deps: [{ token: i1.MetaApiService }, { token: i0.DestroyRef }, { token: i2.EditorialLinkService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
147
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CountsWithHistoryComponent, isStandalone: true, selector: "metaqs2-counts-with-history", inputs: { apiMethod: "apiMethod", columnTranslationkey: "columnTranslationkey", pageTitle: "pageTitle", sourceType: "sourceType" }, ngImport: i0, template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title data-test-id=\"page-title\">\n Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{ isLoading() ? \": Lade neue Daten.\" : \"\" }}\n </mat-card-title>\n </mat-card-header>\n <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <!-- show the filter after the values are loaded to avoid loading current data twice -->\n <mat-card-content class=\"toolbar\">\n <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\n <div style=\"flex: 1 1 auto\"></div>\n <mat-slide-toggle [ngModel]=\"isHistoryEnabled()\" (ngModelChange)=\"isHistoryEnabled.set($event)\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n <label>Zeige historische Daten</label>\n </mat-slide-toggle>\n </mat-card-content>\n</mat-card>\n<mat-card>\n <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentTypeCount$.value.rows\"\n class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"label-col\" sticky>\n <th [attr.rowspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef>\n <div>Quelle</div>\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n [matTooltip]=\"row.meta.alt_label\"\n class=\"label-col\"\n >\n {{ row.meta.label }}\n </td>\n </ng-container>\n <!-- one column for each type -->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_type'\">\n <th [attr.colspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef [matTooltip]=\"col.label\">\n {{ columnTranslationkey ? (columnTranslationkey + col.label | translate) : col.label }}\n </th>\n </ng-container>\n <!-- one column for each type for the most current date-->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_recent'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef\n matTooltip=\"no tooltip\">{{ range.controls.end.value.toLocaleString(DateTime.DATE_SHORT) }}\n </th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\">\n <a [attr.href]=\"openInEditor(row.meta.id, col.id) | async\" target=\"editor_frontend\">{{ row.counts[col.id] ?? '\u2013' }}</a>\n </td>\n </ng-container>\n <!-- one column for each type for the older date-->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_past'\">\n <th class=\"past-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\">\n {{ (pastTypeCount$ | async)?.rows?.length ? range.controls.start.value.toLocaleString(DateTime.DATE_SHORT) : 'no past data' }}\n </th>\n <td class=\"past-data-cell\" mat-cell *matCellDef=\"let row;\" >\n <ng-container *ngIf=\"(pastTypeCount$ | async)?.rows?.length && pastTypeCount(row, col.id) as trend\">\n <span [class]=\"trend.trend\"> {{ trend.value ?? '\u2013' }}\n <mat-icon *ngIf=\"trend.value\" aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" /></span>\n <span class=\"cdk-visually-hidden\">{{ trend.trend }}</span>\n </ng-container>\n </td>\n </ng-container>\n <!-- generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat(typeColumns()); sticky:true;\"></tr>\n <tr [hidden]=\"!isHistoryEnabled()\" mat-header-row *matHeaderRowDef=\"allColumns(); sticky: true;\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(allColumns())\"></tr>\n\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}tr:nth-child(2n){background-color:#e4e4e4}tr:nth-child(2n)>td.label-col,tr:nth-child(2n) td.recent-data-cell{border-right:1px solid white}tr:nth-child(odd){background-color:#fff}tr:nth-child(odd)>td.label-col,tr:nth-child(odd) td.recent-data-cell{border-right:1px solid #e4e4e4}td.label-col{text-align:left}.mat-mdc-header-cell,.mat-mdc-cell{text-align:center}.mat-mdc-header-cell a[href],.mat-mdc-cell a[href]{color:var(--mat-table-row-item-label-text-color);cursor:pointer;text-decoration:underline}.mat-mdc-header-cell a[href]:hover,.mat-mdc-cell a[href]:hover{text-decoration:underline}.mat-mdc-header-cell mat-icon,.mat-mdc-cell mat-icon{margin-left:5px;vertical-align:middle}.trending_down{color:#4abeff}.trending_up{color:#c20808}.actionbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}\n"], dependencies: [{ kind: "ngmodule", type: MatCardModule }, { kind: "component", type: i3.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "directive", type: i3.MatCardContent, selector: "mat-card-content" }, { kind: "component", type: i3.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i3.MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "directive", type: MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatTableModule }, { kind: "component", type: i4.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i4.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i4.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i4.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i4.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i4.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i4.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i4.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i4.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i4.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: DatepickerComponent, selector: "metaqs2-datepicker", inputs: ["disabled", "inputGroup"] }, { kind: "component", type: MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: ProgressSpinnerComponent, selector: "metaqs2-progress-spinner", inputs: ["color", "diameter", "strokeWidth", "backdropEnabled", "positionGloballyCenter", "displayProgressSpinner"] }] }); }
|
|
146
148
|
}
|
|
147
149
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, decorators: [{
|
|
148
150
|
type: Component,
|
|
@@ -162,13 +164,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
162
164
|
MatRowDef,
|
|
163
165
|
NgForOf,
|
|
164
166
|
DatepickerComponent,
|
|
165
|
-
NgClass,
|
|
166
167
|
MatIcon,
|
|
167
168
|
NgIf,
|
|
169
|
+
FormsModule,
|
|
170
|
+
MatSlideToggle,
|
|
168
171
|
AsyncPipe,
|
|
169
172
|
ProgressSpinnerComponent,
|
|
170
|
-
], template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title data-test-id=\"page-title\">\n Qualit\u00E4tsmetrik: {{ pageTitle | translate}}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}\n </mat-card-title>\n </mat-card-header>\n <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <!-- show the filter after the values are loaded to avoid loading current data twice -->\n <mat-card-content
|
|
171
|
-
}], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i2.EditorialLinkService }], propDecorators: { apiMethod: [{
|
|
173
|
+
], template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title data-test-id=\"page-title\">\n Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{ isLoading() ? \": Lade neue Daten.\" : \"\" }}\n </mat-card-title>\n </mat-card-header>\n <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <!-- show the filter after the values are loaded to avoid loading current data twice -->\n <mat-card-content class=\"toolbar\">\n <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\n <div style=\"flex: 1 1 auto\"></div>\n <mat-slide-toggle [ngModel]=\"isHistoryEnabled()\" (ngModelChange)=\"isHistoryEnabled.set($event)\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n <label>Zeige historische Daten</label>\n </mat-slide-toggle>\n </mat-card-content>\n</mat-card>\n<mat-card>\n <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentTypeCount$.value.rows\"\n class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"label-col\" sticky>\n <th [attr.rowspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef>\n <div>Quelle</div>\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n [matTooltip]=\"row.meta.alt_label\"\n class=\"label-col\"\n >\n {{ row.meta.label }}\n </td>\n </ng-container>\n <!-- one column for each type -->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_type'\">\n <th [attr.colspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef [matTooltip]=\"col.label\">\n {{ columnTranslationkey ? (columnTranslationkey + col.label | translate) : col.label }}\n </th>\n </ng-container>\n <!-- one column for each type for the most current date-->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_recent'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef\n matTooltip=\"no tooltip\">{{ range.controls.end.value.toLocaleString(DateTime.DATE_SHORT) }}\n </th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\">\n <a [attr.href]=\"openInEditor(row.meta.id, col.id) | async\" target=\"editor_frontend\">{{ row.counts[col.id] ?? '\u2013' }}</a>\n </td>\n </ng-container>\n <!-- one column for each type for the older date-->\n <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n [matColumnDef]=\"col.id + '_past'\">\n <th class=\"past-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\">\n {{ (pastTypeCount$ | async)?.rows?.length ? range.controls.start.value.toLocaleString(DateTime.DATE_SHORT) : 'no past data' }}\n </th>\n <td class=\"past-data-cell\" mat-cell *matCellDef=\"let row;\" >\n <ng-container *ngIf=\"(pastTypeCount$ | async)?.rows?.length && pastTypeCount(row, col.id) as trend\">\n <span [class]=\"trend.trend\"> {{ trend.value ?? '\u2013' }}\n <mat-icon *ngIf=\"trend.value\" aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" /></span>\n <span class=\"cdk-visually-hidden\">{{ trend.trend }}</span>\n </ng-container>\n </td>\n </ng-container>\n <!-- generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat(typeColumns()); sticky:true;\"></tr>\n <tr [hidden]=\"!isHistoryEnabled()\" mat-header-row *matHeaderRowDef=\"allColumns(); sticky: true;\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(allColumns())\"></tr>\n\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}tr:nth-child(2n){background-color:#e4e4e4}tr:nth-child(2n)>td.label-col,tr:nth-child(2n) td.recent-data-cell{border-right:1px solid white}tr:nth-child(odd){background-color:#fff}tr:nth-child(odd)>td.label-col,tr:nth-child(odd) td.recent-data-cell{border-right:1px solid #e4e4e4}td.label-col{text-align:left}.mat-mdc-header-cell,.mat-mdc-cell{text-align:center}.mat-mdc-header-cell a[href],.mat-mdc-cell a[href]{color:var(--mat-table-row-item-label-text-color);cursor:pointer;text-decoration:underline}.mat-mdc-header-cell a[href]:hover,.mat-mdc-cell a[href]:hover{text-decoration:underline}.mat-mdc-header-cell mat-icon,.mat-mdc-cell mat-icon{margin-left:5px;vertical-align:middle}.trending_down{color:#4abeff}.trending_up{color:#c20808}.actionbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}\n"] }]
|
|
174
|
+
}], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i0.DestroyRef }, { type: i2.EditorialLinkService }], propDecorators: { apiMethod: [{
|
|
172
175
|
type: Input,
|
|
173
176
|
args: [{ required: true }]
|
|
174
177
|
}], columnTranslationkey: [{
|
|
@@ -179,4 +182,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
179
182
|
type: Input,
|
|
180
183
|
args: [{ required: true }]
|
|
181
184
|
}] } });
|
|
182
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"counts-with-history.component.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/counts-with-history/counts-with-history.component.ts","../../../../../projects/ng-meta-widgets-lib/src/lib/counts-with-history/counts-with-history.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnDestroy, OnInit, signal } from '@angular/core';\nimport { MatCard, MatCardContent, MatCardHeader, MatCardModule, MatCardTitle } from '@angular/material/card';\nimport { MatTooltip } from '@angular/material/tooltip';\nimport { MatHeaderRow, MatHeaderRowDef, MatRow, MatRowDef, MatTable, MatTableModule } from '@angular/material/table';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { AsyncPipe, NgClass, NgForOf, NgIf } from '@angular/common';\nimport { MetaApiService } from '../meta-api.service';\nimport { Filter, MatrixRowWithCounts, MatrixWithCounts, QualityMatrixHeader } from '../java-api';\nimport { DateTime } from 'luxon';\nimport { finalize, map, skipWhile, take, takeUntil, tap } from 'rxjs/operators';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { DatepickerComponent } from '../components/filter/datepicker/datepicker.component';\nimport { MatIcon } from '@angular/material/icon';\nimport { EditorialLinkService } from '../components/editorial-link-service/editorial-link.service';\nimport { ProgressSpinnerComponent } from '../components/loading_indicator/progress-spinner/progress-spinner.component';\n\n@Component({\n  selector: 'metaqs2-counts-with-history',\n  standalone: true,\n  imports: [\n    MatCardModule,\n    MatCard,\n    MatCardHeader,\n    MatCardTitle,\n    MatCardContent,\n    MatTooltip,\n    MatTableModule,\n    MatTable,\n    TranslateModule,\n    MatHeaderRow,\n    MatHeaderRowDef,\n    MatRow,\n    MatRowDef,\n    NgForOf,\n    DatepickerComponent,\n    NgClass,\n    MatIcon,\n    NgIf,\n    AsyncPipe,\n    ProgressSpinnerComponent,\n  ],\n  templateUrl: './counts-with-history.component.html',\n  styleUrl: './counts-with-history.component.scss',\n})\nexport class CountsWithHistoryComponent implements OnInit, OnDestroy {\n  protected readonly isLoading = signal(true);\n  protected readonly timeFilterLoaded = signal(false);\n  private readonly destroyed$ = new Subject<void>();\n  protected readonly recentTypeCount$ = new BehaviorSubject<MatrixWithCounts>({ columns: [], rows: [] });\n  protected readonly pastTypeCount$ = new BehaviorSubject<MatrixWithCounts>({ columns: [], rows: [] });\n  private columns: QualityMatrixHeader[] = [];\n\n  @Input({ required: true })\n  apiMethod: keyof MetaApiService = 'getMaterialTypeCountsByReplicationSource';\n  @Input()\n  columnTranslationkey: string | null = null;\n  @Input()\n  pageTitle: string;\n  @Input({ required: true })\n  sourceType: 'replicationSource' | 'collection';\n\n  /* In this widget's backend we do have data for today\n   * therefore we set the end date to today and use the timerange() endpoint to only set the start date\n   */\n  readonly range: FormGroup<{\n    start: FormControl<DateTime<boolean>>;\n    end: FormControl<DateTime<boolean>>;\n  }> = new FormGroup({\n    start: new FormControl(),\n    end: new FormControl(DateTime.utc().startOf('day'), { nonNullable: true }),\n  });\n\n  constructor(private readonly metaApi: MetaApiService, private readonly linkService: EditorialLinkService) {}\n  columnIdent(_index: number, col: QualityMatrixHeader) {\n    return col.id;\n  }\n\n  get typeColumns(): string[] {\n    return this.columns.map((c) => c.id + '_type');\n  }\n\n  get recentColumns(): string[] {\n    return this.columns.map((c) => c.id + '_recent');\n  }\n\n  get pastColumns(): string[] {\n    return this.columns.map((c) => c.id + '_past');\n  }\n\n  get allColumns(): string[] {\n    return this.pastColumns.flatMap((e, i) => [e, this.recentColumns[i]]);\n  }\n\n  private registerDateRangeFilter() {\n    /* we cannot use the takeUntilDestroyed() helper here,\n     * because this method is indirectly called in the finalization of the getTimerangeFilter() call\n     */\n    this.range.controls.end.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((date) => {\n      this.getCountByDate(date).subscribe((response) => {\n        this.recentTypeCount$.next(response);\n      });\n    });\n    this.range.controls.start.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((date) => {\n      this.getCountByDate(date).subscribe((response) => {\n        this.pastTypeCount$.next(response);\n      });\n    });\n  }\n\n  private getTimeRangeFilter() {\n    return this.metaApi\n      .getTimerangeFilter()\n      .pipe(\n        finalize(() => {\n          this.timeFilterLoaded.set(true);\n          this.registerDateRangeFilter();\n          this.range.controls.start.reset();\n        })\n      )\n      .subscribe((rangeFilter) => {\n        if (rangeFilter) {\n          const startDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label!, {\n            zone: 'utc',\n          }).startOf('day');\n          //this is to have a default value for the start date => the min date of the range\n          this.range.setControl('start', new FormControl<DateTime>(startDate, { nonNullable: true }));\n        }\n      });\n  }\n\n  private getCountByDate(date: DateTime): Observable<MatrixWithCounts> {\n    this.isLoading.set(true);\n    const filter: Filter = {\n      field: 'asOf',\n      values: [{ id: date.toISO({ includeOffset: false })!, label: '' }],\n    };\n    // type of this.metaApi[this.apiMethod]([filter]) is Observable<MatrixWithCounts>\n    // but typescript does not know that\n    // @ts-ignore\n    return this.metaApi[this.apiMethod]([filter]).pipe(\n      // typesript gets confused here\n      // @ts-ignore\n      finalize(() => this.isLoading.set(false)),\n      map((response: MatrixWithCounts) => {\n        response.rows.sort((a, b) => a.meta.label.localeCompare(b.meta.label));\n        return response;\n      })\n    );\n  }\n\n  protected pastTypeCount(row: MatrixRowWithCounts, columnid: string) {\n    if (!this.pastTypeCount$.value.rows.length) {\n      return {};\n    }\n    const past_row = this.pastTypeCount$.value.rows.find((pr) => pr.meta.id == row.meta.id);\n    const past_column = this.pastTypeCount$.value.columns.find((c) => c.id == columnid);\n    if (!past_row || !past_column) {\n      return {};\n    }\n\n    const currentValue = row.counts[columnid];\n    const pastValue = past_row.counts[columnid];\n\n    const delta = (currentValue || 0) - (pastValue || 0);\n    const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';\n    return { delta, trend, value: pastValue };\n  }\n\n  protected openInEditor(sourceId: string, issueId: string) {\n    if (this.sourceType === 'replicationSource') {\n      // find the long name for the source\n      sourceId = this.recentTypeCount$.value.rows.find((r) => r.meta.id === sourceId)?.meta.label || sourceId;\n    }\n    return this.linkService.typesLoaded$.pipe(\n      skipWhile((loaded) => !loaded),\n      take(1),\n      map(() => {\n        return this.linkService.createLinkForCountsWithHistory(this.sourceType, sourceId, issueId);\n      })\n    );\n  }\n\n  ngOnInit(): void {\n    this.getTimeRangeFilter();\n    this.getCountByDate(this.range.controls.end.value)\n      .pipe(\n        tap((response) => {\n          this.columns = response.columns.slice();\n        })\n      )\n      .subscribe((response) => {\n        this.recentTypeCount$.next(response);\n      });\n  }\n\n  protected readonly DateTime = DateTime;\n\n  ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n}\n","<mat-card appearance=\"raised\">\n  <mat-card-header *ngIf=\"pageTitle\">\n    <mat-card-title data-test-id=\"page-title\">\n      Qualitätsmetrik: {{ pageTitle | translate}}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}\n    </mat-card-title>\n  </mat-card-header>\n  <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n  <!-- show the filter after the values are loaded to avoid loading current data twice -->\n  <mat-card-content *ngIf=\"timeFilterLoaded()\">\n    <metaqs2-datepicker [disabled]=\"isLoading()\" [inputGroup]=\"range\" ></metaqs2-datepicker>\n  </mat-card-content>\n</mat-card>\n<mat-card>\n  <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n  <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"recentTypeCount$.value.rows\" class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Row Header Column -->\n    <ng-container matColumnDef=\"label-col\" sticky>\n      <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Quelle</th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        matTooltip=\"{{row.meta.alt_label}}\"\n        class=\"label-col\"\n      >\n        {{row.meta.label}}\n      </td>\n    </ng-container>\n    <!-- one column for each type -->\n    <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n      <th colspan=\"2\" mat-header-cell *matHeaderCellDef matTooltip=\"{{col.label}}\" >\n        {{columnTranslationkey ? (columnTranslationkey + col.label | translate ) : col.label }}\n      </th>\n    </ng-container>\n    <!-- one column for each type for the most current date-->\n    <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_recent'\">\n      <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\" >{{ range.controls.end.value.toLocaleString(DateTime.DATE_SHORT) }}</th>\n      <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\">\n        <a [attr.href]=\"openInEditor(row.meta.id, col.id) | async\" target=\"editor_frontend\">{{ row.counts[col.id] ?? 'n/a' }}</a>\n      </td>\n    </ng-container>\n    <!-- one column for each type for the older date-->\n    <ng-container *ngFor=\"let col of recentTypeCount$.value.columns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_past'\">\n      <th class=\"past-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\" >\n        {{ pastTypeCount$.value.rows.length ? range.controls.start.value.toLocaleString(DateTime.DATE_SHORT) : 'no past data' }}\n      </th>\n      <td class=\"past-data-cell\" mat-cell *matCellDef=\"let row;\">\n        <ng-container *ngIf=\"pastTypeCount$.value.rows.length && pastTypeCount(row, col.id) as trend\">\n          <span [ngClass]=\"trend.trend\"> {{ trend.value ?? 'n/a' }}<mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" /></span>\n          <span class=\"cdk-visually-hidden\">{{trend.trend}}</span>\n        </ng-container>\n      </td>\n    </ng-container>\n    <!-- generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat(typeColumns); sticky:true;\"></tr>\n    <tr mat-header-row *matHeaderRowDef=\"allColumns; sticky: true;\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(allColumns)\"></tr>\n\n  </table>\n</mat-card>"]}
|
|
185
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"counts-with-history.component.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/counts-with-history/counts-with-history.component.ts","../../../../../projects/ng-meta-widgets-lib/src/lib/counts-with-history/counts-with-history.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, DestroyRef, effect, Input, OnInit, signal } from '@angular/core';\nimport { MatCard, MatCardContent, MatCardHeader, MatCardModule, MatCardTitle } from '@angular/material/card';\nimport { MatTooltip } from '@angular/material/tooltip';\nimport { MatHeaderRow, MatHeaderRowDef, MatRow, MatRowDef, MatTable, MatTableModule } from '@angular/material/table';\nimport { BehaviorSubject, Observable, zip } from 'rxjs';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { AsyncPipe, NgForOf, NgIf } from '@angular/common';\nimport { MetaApiService } from '../meta-api.service';\nimport { Filter, MatrixRowWithCounts, MatrixWithCounts, QualityMatrixHeader } from '../java-api';\nimport { DateTime } from 'luxon';\nimport { filter, finalize, map, skipWhile, switchMap, take, tap } from 'rxjs/operators';\nimport { FormControl, FormGroup, FormsModule } from '@angular/forms';\nimport { DatepickerComponent } from '../components/filter/datepicker/datepicker.component';\nimport { MatIcon } from '@angular/material/icon';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { MatSlideToggle } from '@angular/material/slide-toggle';\nimport { ProgressSpinnerComponent } from '../components/loading_indicator/progress-spinner/progress-spinner.component';\nimport { EditorialLinkService } from '../components/editorial-link-service/editorial-link.service';\n\ntype PickKeysByPropertyType<TObject extends object, TPropertyType> = {\n  [TKey in keyof TObject]: TObject[TKey] extends TPropertyType ? TKey : never;\n}[keyof TObject];\n\n@Component({\n  selector: 'metaqs2-counts-with-history',\n  standalone: true,\n  imports: [\n    MatCardModule,\n    MatCard,\n    MatCardHeader,\n    MatCardTitle,\n    MatCardContent,\n    MatTooltip,\n    MatTableModule,\n    MatTable,\n    TranslateModule,\n    MatHeaderRow,\n    MatHeaderRowDef,\n    MatRow,\n    MatRowDef,\n    NgForOf,\n    DatepickerComponent,\n    MatIcon,\n    NgIf,\n    FormsModule,\n    MatSlideToggle,\n    AsyncPipe,\n    ProgressSpinnerComponent,\n  ],\n  templateUrl: './counts-with-history.component.html',\n  styleUrl: './counts-with-history.component.scss',\n})\nexport class CountsWithHistoryComponent implements OnInit {\n  protected readonly loadingCount = signal(0);\n  protected readonly isLoading = computed(() => this.loadingCount() > 0);\n  protected readonly timeFilterLoaded = signal(false);\n  protected readonly recentTypeCount$ = new BehaviorSubject<MatrixWithCounts>({ columns: [], rows: [] });\n  protected readonly pastTypeCount$ = new BehaviorSubject<MatrixWithCounts>({ columns: [], rows: [] });\n  protected readonly DateTime = DateTime;\n  public columns = signal([] as Array<QualityMatrixHeader>);\n\n  @Input({ required: true })\n  apiMethod: PickKeysByPropertyType<MetaApiService, (it: Array<Filter>) => Observable<MatrixWithCounts>> =\n    'getMaterialTypeCountsByReplicationSource';\n  @Input()\n  columnTranslationkey: string | null = null;\n  @Input()\n  pageTitle: string;\n  @Input({ required: true })\n  sourceType: 'replicationSource' | 'collection';\n\n  /* In this widget's backend we do have data for today\n   * therefore we set the end date to today and use the timerange() endpoint to only set the start date\n   */\n  readonly range: FormGroup<{\n    start: FormControl<DateTime<boolean>>;\n    end: FormControl<DateTime<boolean>>;\n  }> = new FormGroup({\n    start: new FormControl(),\n    end: new FormControl(DateTime.utc().startOf('day'), { nonNullable: true }),\n  });\n  readonly isHistoryEnabled = signal<boolean>(false);\n\n  readonly allColumns = computed<Array<string>>(() => {\n    if (!this.isHistoryEnabled()) {\n      return this.recentColumns();\n    }\n    return this.pastColumns().flatMap((e, i) => [e, this.recentColumns()[i]]);\n  });\n\n  readonly typeColumns = computed(() => {\n    return this.columns().map((c) => c.id + '_type');\n  });\n\n  readonly recentColumns = computed(() => {\n    return this.columns().map((c) => c.id + '_recent');\n  });\n\n  readonly pastColumns = computed(() => {\n    return this.columns().map((c) => c.id + '_past');\n  });\n\n  constructor(\n    protected readonly metaApi: MetaApiService,\n    protected readonly destroyRef: DestroyRef,\n    private readonly linkService: EditorialLinkService\n  ) {\n    effect(() => {\n      this.range.controls.end.reset();\n      if (!this.isHistoryEnabled()) {\n        this.range.controls.start.setValue(this.range.controls.end.value.startOf('day'));\n      } else {\n        this.range.controls.start.reset();\n      }\n    });\n  }\n\n  ngOnInit(): void {\n    this.getAvailableDateRange()\n      .pipe(\n        tap((rangeFilter) => {\n          const startDate = this.getStartDateOfRange(rangeFilter);\n          //this is to have a default value for the start date => the min date of the range\n          this.range.setControl('start', new FormControl<DateTime<true>>(startDate, { nonNullable: true }));\n        }),\n        finalize(() => {\n          this.timeFilterLoaded.set(true);\n          this.range.controls.start.reset();\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n    this.getCountByDate(this.range.controls.end.value)\n      .pipe(\n        tap((response) => {\n          this.columns.set(response.columns.slice());\n          this.recentTypeCount$.next(response);\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n    this.getValuesInDateRange()\n      .pipe(\n        tap(([past, recent]) => {\n          this.pastTypeCount$.next(past);\n          this.recentTypeCount$.next(recent);\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n  }\n\n  private getStartDateOfRange(rangeFilter: Filter) {\n    return DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label!, {\n      zone: 'utc',\n    }).startOf('day') as DateTime<true>;\n  }\n\n  columnIdent(_index: number, col: QualityMatrixHeader) {\n    return col.id;\n  }\n\n  private getAvailableDateRange() {\n    this.loadingCount.update((it) => it + 1);\n    return this.metaApi.getTimerangeFilter().pipe(\n      filter((filter): filter is Filter => filter != null),\n      finalize(() => this.loadingCount.update((it) => it - 1))\n    );\n  }\n\n  private getValuesInDateRange() {\n    return this.range.valueChanges.pipe(\n      filter(\n        (range): range is { start: DateTime<true>; end: DateTime<true> } =>\n          !!range.start?.isValid && !!range.end?.isValid\n      ),\n      switchMap((range) => {\n        return zip(this.getCountByDate(range.start), this.getCountByDate(range.end));\n      })\n    );\n  }\n\n  private getCountByDate(date: DateTime): Observable<MatrixWithCounts> {\n    this.loadingCount.update((it) => it + 1);\n    const filter: Filter = {\n      field: 'asOf',\n      values: [{ id: date.toISO({ includeOffset: false })!, label: '' }],\n    };\n    return this.metaApi[this.apiMethod]([filter]).pipe(\n      finalize(() => this.loadingCount.update((it) => it - 1)),\n      map((response) => ({\n        ...response,\n        rows: response.rows.toSorted((a, b) => a.meta.label.localeCompare(b.meta.label)),\n      }))\n    );\n  }\n\n  protected pastTypeCount(\n    row: MatrixRowWithCounts,\n    columnid: string\n  ): { delta?: number; trend?: string; value?: number } {\n    if (!this.pastTypeCount$.value.rows.length) {\n      return {};\n    }\n    const past_row = this.pastTypeCount$.value.rows.find((pr) => pr.meta.id == row.meta.id);\n    const past_column = this.pastTypeCount$.value.columns.find((c) => c.id == columnid);\n    if (!past_row || !past_column) {\n      return {};\n    }\n\n    const currentValue = row.counts[columnid];\n    const pastValue = past_row.counts[columnid];\n\n    const delta = (currentValue || 0) - (pastValue || 0);\n    const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';\n    return { delta, trend, value: pastValue };\n  }\n\n  protected openInEditor(sourceId: string, issueId: string) {\n    if (this.sourceType === 'replicationSource') {\n      // find the long name for the source\n      sourceId = this.recentTypeCount$.value.rows.find((r) => r.meta.id === sourceId)?.meta.label || sourceId;\n    }\n    return this.linkService.typesLoaded$.pipe(\n      skipWhile((loaded) => !loaded),\n      take(1),\n      map(() => {\n        return this.linkService.createLinkForCountsWithHistory(this.sourceType, sourceId, issueId);\n      })\n    );\n  }\n}\n","<mat-card appearance=\"raised\">\n  <mat-card-header *ngIf=\"pageTitle\">\n    <mat-card-title data-test-id=\"page-title\">\n      Qualitätsmetrik: {{ pageTitle | translate }}{{ isLoading() ? \": Lade neue Daten.\" : \"\" }}\n    </mat-card-title>\n  </mat-card-header>\n  <!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n  <!-- show the filter after the values are loaded to avoid loading current data twice -->\n  <mat-card-content class=\"toolbar\">\n    <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\n    <div style=\"flex: 1 1 auto\"></div>\n    <mat-slide-toggle [ngModel]=\"isHistoryEnabled()\" (ngModelChange)=\"isHistoryEnabled.set($event)\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n      <label>Zeige historische Daten</label>\n    </mat-slide-toggle>\n  </mat-card-content>\n</mat-card>\n<mat-card>\n  <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n  <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentTypeCount$.value.rows\"\n         class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Row Header Column -->\n    <ng-container matColumnDef=\"label-col\" sticky>\n      <th [attr.rowspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef>\n        <div>Quelle</div>\n      </th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        [matTooltip]=\"row.meta.alt_label\"\n        class=\"label-col\"\n      >\n        {{ row.meta.label }}\n      </td>\n    </ng-container>\n    <!-- one column for each type -->\n    <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n                  [matColumnDef]=\"col.id + '_type'\">\n      <th [attr.colspan]=\"isHistoryEnabled() ? '2' : '1'\" mat-header-cell *matHeaderCellDef [matTooltip]=\"col.label\">\n        {{ columnTranslationkey ? (columnTranslationkey + col.label | translate) : col.label }}\n      </th>\n    </ng-container>\n    <!-- one column for each type for the most current date-->\n    <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n                  [matColumnDef]=\"col.id + '_recent'\">\n      <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef\n          matTooltip=\"no tooltip\">{{ range.controls.end.value.toLocaleString(DateTime.DATE_SHORT) }}\n      </th>\n      <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\">\n        <a [attr.href]=\"openInEditor(row.meta.id, col.id) | async\" target=\"editor_frontend\">{{ row.counts[col.id] ?? '–' }}</a>\n      </td>\n    </ng-container>\n    <!-- one column for each type for the older date-->\n    <ng-container *ngFor=\"let col of (recentTypeCount$ | async)?.columns; trackBy:columnIdent\"\n                  [matColumnDef]=\"col.id + '_past'\">\n      <th class=\"past-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"no tooltip\">\n        {{ (pastTypeCount$ | async)?.rows?.length ? range.controls.start.value.toLocaleString(DateTime.DATE_SHORT) : 'no past data' }}\n      </th>\n      <td class=\"past-data-cell\" mat-cell *matCellDef=\"let row;\" >\n        <ng-container *ngIf=\"(pastTypeCount$ | async)?.rows?.length && pastTypeCount(row, col.id) as trend\">\n          <span [class]=\"trend.trend\"> {{ trend.value ?? '–' }}\n            <mat-icon *ngIf=\"trend.value\" aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" /></span>\n          <span class=\"cdk-visually-hidden\">{{ trend.trend }}</span>\n        </ng-container>\n      </td>\n    </ng-container>\n    <!-- generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat(typeColumns()); sticky:true;\"></tr>\n    <tr [hidden]=\"!isHistoryEnabled()\" mat-header-row *matHeaderRowDef=\"allColumns(); sticky: true;\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(allColumns())\"></tr>\n\n  </table>\n</mat-card>\n"]}
|