ngx-edu-sharing-metaqs2 0.9.31 → 0.9.33
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/collection-count-history/collection-count-history.component.mjs +48 -32
- package/esm2022/lib/collection-count-history/monthpicker/monthpicker.component.mjs +7 -4
- package/esm2022/lib/components/collection-issues/collection-issues.component.mjs +2 -2
- package/esm2022/lib/components/donut-chart/donut-chart.component.mjs +9 -9
- package/esm2022/lib/components/filter/datepicker/datepicker.component.mjs +3 -3
- package/esm2022/lib/components/loading_indicator/overlay/overlay.service.mjs +41 -0
- package/esm2022/lib/components/loading_indicator/progress-spinner/progress-spinner.component.mjs +65 -0
- package/esm2022/lib/components/material-issues/material-issues.component.mjs +2 -2
- package/esm2022/lib/components/node-entry/node-entry.component.mjs +2 -2
- package/esm2022/lib/components/node-list/node-list.component.mjs +2 -2
- package/esm2022/lib/components/quality-matrix/quality_matrix.mjs +6 -5
- package/esm2022/lib/counts-with-history/counts-with-history.component.mjs +89 -84
- package/esm2022/lib/ng-meta-widgets-lib.module.mjs +8 -4
- package/esm2022/lib/tree-collection-details/tree-collection-details.component.mjs +17 -3
- package/esm2022/lib/tree-search-counts/tree-search-counts.component.mjs +5 -3
- package/esm2022/public-api.mjs +1 -2
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs +277 -277
- 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/loading_indicator/overlay/overlay.service.d.ts +13 -0
- package/lib/components/loading_indicator/progress-spinner/progress-spinner.component.d.ts +21 -0
- package/lib/counts-with-history/counts-with-history.component.d.ts +27 -25
- package/lib/ng-meta-widgets-lib.module.d.ts +2 -1
- package/package.json +1 -1
- package/public-api.d.ts +0 -1
- 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,31 +1,37 @@
|
|
|
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';
|
|
15
|
+
import { ProgressSpinnerComponent } from '../components/loading_indicator/progress-spinner/progress-spinner.component';
|
|
13
16
|
import * as i0 from "@angular/core";
|
|
14
17
|
import * as i1 from "../meta-api.service";
|
|
15
18
|
import * as i2 from "../components/editorial-link-service/editorial-link.service";
|
|
16
19
|
import * as i3 from "@angular/material/card";
|
|
17
20
|
import * as i4 from "@angular/material/table";
|
|
18
21
|
import * as i5 from "@ngx-translate/core";
|
|
22
|
+
import * as i6 from "@angular/forms";
|
|
19
23
|
export class CountsWithHistoryComponent {
|
|
20
|
-
constructor(metaApi, linkService) {
|
|
24
|
+
constructor(metaApi, destroyRef, linkService) {
|
|
21
25
|
this.metaApi = metaApi;
|
|
26
|
+
this.destroyRef = destroyRef;
|
|
22
27
|
this.linkService = linkService;
|
|
23
|
-
this.
|
|
28
|
+
this.loadingCount = signal(0);
|
|
29
|
+
this.isLoading = computed(() => this.loadingCount() > 0);
|
|
24
30
|
this.timeFilterLoaded = signal(false);
|
|
25
|
-
this.destroyed$ = new Subject();
|
|
26
31
|
this.recentTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
27
32
|
this.pastTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
28
|
-
this.
|
|
33
|
+
this.DateTime = DateTime;
|
|
34
|
+
this.columns = signal([]);
|
|
29
35
|
this.apiMethod = 'getMaterialTypeCountsByReplicationSource';
|
|
30
36
|
this.columnTranslationkey = null;
|
|
31
37
|
/* In this widget's backend we do have data for today
|
|
@@ -35,72 +41,83 @@ export class CountsWithHistoryComponent {
|
|
|
35
41
|
start: new FormControl(),
|
|
36
42
|
end: new FormControl(DateTime.utc().startOf('day'), { nonNullable: true }),
|
|
37
43
|
});
|
|
38
|
-
this.
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
return this.columns.map((c) => c.id + '_recent');
|
|
48
|
-
}
|
|
49
|
-
get pastColumns() {
|
|
50
|
-
return this.columns.map((c) => c.id + '_past');
|
|
51
|
-
}
|
|
52
|
-
get allColumns() {
|
|
53
|
-
return this.pastColumns.flatMap((e, i) => [e, this.recentColumns[i]]);
|
|
54
|
-
}
|
|
55
|
-
registerDateRangeFilter() {
|
|
56
|
-
/* we cannot use the takeUntilDestroyed() helper here,
|
|
57
|
-
* because this method is indirectly called in the finalization of the getTimerangeFilter() call
|
|
58
|
-
*/
|
|
59
|
-
this.range.controls.end.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((date) => {
|
|
60
|
-
this.getCountByDate(date).subscribe((response) => {
|
|
61
|
-
this.recentTypeCount$.next(response);
|
|
62
|
-
});
|
|
44
|
+
this.isHistoryEnabled = signal(true);
|
|
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');
|
|
63
53
|
});
|
|
64
|
-
this.
|
|
65
|
-
this.
|
|
66
|
-
|
|
67
|
-
|
|
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
|
+
}
|
|
68
68
|
});
|
|
69
69
|
}
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
.
|
|
73
|
-
.
|
|
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(() => {
|
|
74
77
|
this.timeFilterLoaded.set(true);
|
|
75
|
-
this.registerDateRangeFilter();
|
|
76
78
|
this.range.controls.start.reset();
|
|
77
|
-
}))
|
|
78
|
-
.subscribe(
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
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
|
+
}));
|
|
87
110
|
}
|
|
88
111
|
getCountByDate(date) {
|
|
89
|
-
this.
|
|
112
|
+
this.loadingCount.update((it) => it + 1);
|
|
90
113
|
const filter = {
|
|
91
114
|
field: 'asOf',
|
|
92
115
|
values: [{ id: date.toISO({ includeOffset: false }), label: '' }],
|
|
93
116
|
};
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
// typesript gets confused here
|
|
99
|
-
// @ts-ignore
|
|
100
|
-
finalize(() => this.isLoading.set(false)), map((response) => {
|
|
101
|
-
response.rows.sort((a, b) => a.meta.label.localeCompare(b.meta.label));
|
|
102
|
-
return response;
|
|
103
|
-
}));
|
|
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
|
+
})));
|
|
104
121
|
}
|
|
105
122
|
pastTypeCount(row, columnid) {
|
|
106
123
|
if (!this.pastTypeCount$.value.rows.length) {
|
|
@@ -126,22 +143,8 @@ export class CountsWithHistoryComponent {
|
|
|
126
143
|
return this.linkService.createLinkForCountsWithHistory(this.sourceType, sourceId, issueId);
|
|
127
144
|
}));
|
|
128
145
|
}
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
this.getCountByDate(this.range.controls.end.value)
|
|
132
|
-
.pipe(tap((response) => {
|
|
133
|
-
this.columns = response.columns.slice();
|
|
134
|
-
}))
|
|
135
|
-
.subscribe((response) => {
|
|
136
|
-
this.recentTypeCount$.next(response);
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
ngOnDestroy() {
|
|
140
|
-
this.destroyed$.next();
|
|
141
|
-
this.destroyed$.complete();
|
|
142
|
-
}
|
|
143
|
-
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 }); }
|
|
144
|
-
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 <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: ["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" }] }); }
|
|
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>\n <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded()\"></metaqs2-datepicker>\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 <mat-slide-toggle [(ngModel)]=\"isHistoryEnabled\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n <label>Zeige historische Daten</label>\n </mat-slide-toggle>\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}\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"] }] }); }
|
|
145
148
|
}
|
|
146
149
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, decorators: [{
|
|
147
150
|
type: Component,
|
|
@@ -161,12 +164,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
161
164
|
MatRowDef,
|
|
162
165
|
NgForOf,
|
|
163
166
|
DatepickerComponent,
|
|
164
|
-
NgClass,
|
|
165
167
|
MatIcon,
|
|
166
168
|
NgIf,
|
|
169
|
+
FormsModule,
|
|
170
|
+
MatSlideToggle,
|
|
167
171
|
AsyncPipe,
|
|
168
|
-
|
|
169
|
-
|
|
172
|
+
ProgressSpinnerComponent,
|
|
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>\n <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded()\"></metaqs2-datepicker>\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 <mat-slide-toggle [(ngModel)]=\"isHistoryEnabled\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n <label>Zeige historische Daten</label>\n </mat-slide-toggle>\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}\n"] }]
|
|
174
|
+
}], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i0.DestroyRef }, { type: i2.EditorialLinkService }], propDecorators: { apiMethod: [{
|
|
170
175
|
type: Input,
|
|
171
176
|
args: [{ required: true }]
|
|
172
177
|
}], columnTranslationkey: [{
|
|
@@ -177,4 +182,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
177
182
|
type: Input,
|
|
178
183
|
args: [{ required: true }]
|
|
179
184
|
}] } });
|
|
180
|
-
//# 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';\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  ],\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  <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>(true);\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>\n    <metaqs2-datepicker [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded()\"></metaqs2-datepicker>\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        <mat-slide-toggle [(ngModel)]=\"isHistoryEnabled\" [disabled]=\"isLoading()\" *ngIf=\"timeFilterLoaded()\">\n          <label>Zeige historische Daten</label>\n        </mat-slide-toggle>\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"]}
|
|
@@ -36,6 +36,7 @@ import { NodeEntryComponent } from './components/node-entry/node-entry.component
|
|
|
36
36
|
import { NodeImageUrlPipe } from './node-image-url.pipe';
|
|
37
37
|
import { MaterialIssuesComponent } from './components/material-issues/material-issues.component';
|
|
38
38
|
import { CollectionIssuesComponent } from './components/collection-issues/collection-issues.component';
|
|
39
|
+
import { ProgressSpinnerComponent } from './components/loading_indicator/progress-spinner/progress-spinner.component';
|
|
39
40
|
import * as i0 from "@angular/core";
|
|
40
41
|
import * as i1 from "./core/svg-icons.service";
|
|
41
42
|
import * as i2 from "./java-api/api.module";
|
|
@@ -100,7 +101,8 @@ export class NgMetaWidgetsLibModule {
|
|
|
100
101
|
MatDateRangePicker,
|
|
101
102
|
MatDatepickerInput,
|
|
102
103
|
MatDatepicker,
|
|
103
|
-
DatepickerComponent
|
|
104
|
+
DatepickerComponent,
|
|
105
|
+
ProgressSpinnerComponent], exports: [CollectionIssuesComponent, MaterialIssuesComponent, NodeListComponent, QualityMatrixComponent] }); }
|
|
104
106
|
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgMetaWidgetsLibModule, providers: [provideHttpClient(withInterceptorsFromDi()), provideCharts(withDefaultRegisterables())], imports: [BrowserModule,
|
|
105
107
|
JavaApiModule.forRoot(() => new JavaConfig()),
|
|
106
108
|
MatCardModule,
|
|
@@ -130,7 +132,8 @@ export class NgMetaWidgetsLibModule {
|
|
|
130
132
|
MatDatepickerModule,
|
|
131
133
|
MatDateRangePicker,
|
|
132
134
|
MatDatepicker,
|
|
133
|
-
DatepickerComponent
|
|
135
|
+
DatepickerComponent,
|
|
136
|
+
ProgressSpinnerComponent] }); }
|
|
134
137
|
}
|
|
135
138
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgMetaWidgetsLibModule, decorators: [{
|
|
136
139
|
type: NgModule,
|
|
@@ -146,7 +149,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
146
149
|
NodeListComponent,
|
|
147
150
|
NodeImageUrlPipe,
|
|
148
151
|
],
|
|
149
|
-
exports: [CollectionIssuesComponent, MaterialIssuesComponent, NodeListComponent],
|
|
152
|
+
exports: [CollectionIssuesComponent, MaterialIssuesComponent, NodeListComponent, QualityMatrixComponent],
|
|
150
153
|
imports: [
|
|
151
154
|
BrowserModule,
|
|
152
155
|
JavaApiModule.forRoot(() => new JavaConfig()),
|
|
@@ -180,8 +183,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
180
183
|
MatDatepickerInput,
|
|
181
184
|
MatDatepicker,
|
|
182
185
|
DatepickerComponent,
|
|
186
|
+
ProgressSpinnerComponent,
|
|
183
187
|
],
|
|
184
188
|
providers: [provideHttpClient(withInterceptorsFromDi()), provideCharts(withDefaultRegisterables())],
|
|
185
189
|
}]
|
|
186
190
|
}], ctorParameters: () => [{ type: i1.SvgIconsService }] });
|
|
187
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-meta-widgets-lib.module.js","sourceRoot":"","sources":["../../../../projects/ng-meta-widgets-lib/src/lib/ng-meta-widgets-lib.module.ts"],"names":[],"mappings":"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","sourcesContent":["import { ModuleWithProviders, NgModule } from '@angular/core';\nimport { MatCardModule } from '@angular/material/card';\nimport { MatTreeModule } from '@angular/material/tree';\nimport { BrowserAnimationsModule } from '@angular/platform-browser/animations';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatSliderModule } from '@angular/material/slider';\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTableModule } from '@angular/material/table';\nimport { MatGridListModule } from '@angular/material/grid-list';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSidenavModule } from '@angular/material/sidenav';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { ApiModule as JavaApiModule, Configuration as JavaConfig } from './java-api';\nimport { WrapObservablePipe } from './wrap-observable.pipe';\nimport { ConfigHelperService, NgMetaWidgetsLibConfiguration } from './config-helper.service';\nimport { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http';\nimport { BrowserModule } from '@angular/platform-browser';\nimport { MatListModule } from '@angular/material/list';\nimport { MatSort, MatSortModule } from '@angular/material/sort';\nimport { MatCheckbox } from '@angular/material/checkbox';\nimport { QualityMatrixFilterComponent } from './components/filter/quality-matrix-filter.component';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { QualityMatrixComponent } from './components/quality-matrix/quality_matrix';\nimport {\n  MatDatepicker,\n  MatDatepickerInput,\n  MatDatepickerModule,\n  MatDatepickerToggle,\n  MatDateRangeInput,\n  MatDateRangePicker,\n} from '@angular/material/datepicker';\nimport { DatepickerComponent } from './components/filter/datepicker/datepicker.component';\nimport { DonutChartComponent } from './components/donut-chart/donut-chart.component';\nimport { DonutChartPipe } from './components/donut-chart/donut-chart.pipe';\nimport { provideCharts, withDefaultRegisterables } from 'ng2-charts';\nimport { SvgIconsService } from './core/svg-icons.service';\nimport { NodeListComponent } from './components/node-list/node-list.component';\nimport { NodeEntryComponent } from './components/node-entry/node-entry.component';\nimport { NodeImageUrlPipe } from './node-image-url.pipe';\nimport { MaterialIssuesComponent } from './components/material-issues/material-issues.component';\nimport { CollectionIssuesComponent } from './components/collection-issues/collection-issues.component';\n\n@NgModule({\n  declarations: [\n    WrapObservablePipe,\n    QualityMatrixComponent,\n    DonutChartComponent,\n    DonutChartPipe,\n    CollectionIssuesComponent,\n    MaterialIssuesComponent,\n    NodeEntryComponent,\n    NodeListComponent,\n    NodeImageUrlPipe,\n  ],\n  exports: [CollectionIssuesComponent, MaterialIssuesComponent, NodeListComponent],\n  imports: [\n    BrowserModule,\n    JavaApiModule.forRoot(() => new JavaConfig()),\n    MatCardModule,\n    MatTreeModule,\n    BrowserAnimationsModule,\n    MatIconModule,\n    MatSliderModule,\n    MatSlideToggleModule,\n    MatButtonModule,\n    MatProgressSpinnerModule,\n    MatTableModule,\n    MatGridListModule,\n    MatFormFieldModule,\n    MatInputModule,\n    MatSidenavModule,\n    FormsModule,\n    MatSelectModule,\n    ReactiveFormsModule,\n    MatTooltipModule,\n    MatListModule,\n    MatSort,\n    MatSortModule,\n    MatCheckbox,\n    TranslateModule,\n    QualityMatrixFilterComponent,\n    MatDateRangeInput,\n    MatDatepickerToggle,\n    MatDatepickerModule,\n    MatDateRangePicker,\n    MatDatepickerInput,\n    MatDatepicker,\n    DatepickerComponent,\n  ],\n  providers: [provideHttpClient(withInterceptorsFromDi()), provideCharts(withDefaultRegisterables())],\n})\nexport class NgMetaWidgetsLibModule {\n  constructor(private readonly svgIconsService: SvgIconsService) {\n    // we inject the service here to make sure the icons are registered (the init() method is called in the constructor)\n  }\n  static forRoot(params: NgMetaWidgetsLibConfiguration): ModuleWithProviders<NgMetaWidgetsLibModule> {\n    return {\n      ngModule: NgMetaWidgetsLibModule,\n      providers: [\n        ConfigHelperService,\n        { provide: 'config', useValue: params },\n        {\n          provide: JavaConfig,\n          useFactory: () => {\n            return new JavaConfig({\n              basePath: params.apiPath,\n              withCredentials: true,\n            });\n          },\n        },\n      ],\n    };\n  }\n}\n"]}
|
|
191
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-meta-widgets-lib.module.js","sourceRoot":"","sources":["../../../../projects/ng-meta-widgets-lib/src/lib/ng-meta-widgets-lib.module.ts"],"names":[],"mappings":"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","sourcesContent":["import { ModuleWithProviders, NgModule } from '@angular/core';\nimport { MatCardModule } from '@angular/material/card';\nimport { MatTreeModule } from '@angular/material/tree';\nimport { BrowserAnimationsModule } from '@angular/platform-browser/animations';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatSliderModule } from '@angular/material/slider';\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { MatTableModule } from '@angular/material/table';\nimport { MatGridListModule } from '@angular/material/grid-list';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSidenavModule } from '@angular/material/sidenav';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { ApiModule as JavaApiModule, Configuration as JavaConfig } from './java-api';\nimport { WrapObservablePipe } from './wrap-observable.pipe';\nimport { ConfigHelperService, NgMetaWidgetsLibConfiguration } from './config-helper.service';\nimport { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http';\nimport { BrowserModule } from '@angular/platform-browser';\nimport { MatListModule } from '@angular/material/list';\nimport { MatSort, MatSortModule } from '@angular/material/sort';\nimport { MatCheckbox } from '@angular/material/checkbox';\nimport { QualityMatrixFilterComponent } from './components/filter/quality-matrix-filter.component';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { QualityMatrixComponent } from './components/quality-matrix/quality_matrix';\nimport {\n  MatDatepicker,\n  MatDatepickerInput,\n  MatDatepickerModule,\n  MatDatepickerToggle,\n  MatDateRangeInput,\n  MatDateRangePicker,\n} from '@angular/material/datepicker';\nimport { DatepickerComponent } from './components/filter/datepicker/datepicker.component';\nimport { DonutChartComponent } from './components/donut-chart/donut-chart.component';\nimport { DonutChartPipe } from './components/donut-chart/donut-chart.pipe';\nimport { provideCharts, withDefaultRegisterables } from 'ng2-charts';\nimport { SvgIconsService } from './core/svg-icons.service';\nimport { NodeListComponent } from './components/node-list/node-list.component';\nimport { NodeEntryComponent } from './components/node-entry/node-entry.component';\nimport { NodeImageUrlPipe } from './node-image-url.pipe';\nimport { MaterialIssuesComponent } from './components/material-issues/material-issues.component';\nimport { CollectionIssuesComponent } from './components/collection-issues/collection-issues.component';\nimport { ProgressSpinnerComponent } from './components/loading_indicator/progress-spinner/progress-spinner.component';\n\n@NgModule({\n  declarations: [\n    WrapObservablePipe,\n    QualityMatrixComponent,\n    DonutChartComponent,\n    DonutChartPipe,\n    CollectionIssuesComponent,\n    MaterialIssuesComponent,\n    NodeEntryComponent,\n    NodeListComponent,\n    NodeImageUrlPipe,\n  ],\n  exports: [CollectionIssuesComponent, MaterialIssuesComponent, NodeListComponent, QualityMatrixComponent],\n  imports: [\n    BrowserModule,\n    JavaApiModule.forRoot(() => new JavaConfig()),\n    MatCardModule,\n    MatTreeModule,\n    BrowserAnimationsModule,\n    MatIconModule,\n    MatSliderModule,\n    MatSlideToggleModule,\n    MatButtonModule,\n    MatProgressSpinnerModule,\n    MatTableModule,\n    MatGridListModule,\n    MatFormFieldModule,\n    MatInputModule,\n    MatSidenavModule,\n    FormsModule,\n    MatSelectModule,\n    ReactiveFormsModule,\n    MatTooltipModule,\n    MatListModule,\n    MatSort,\n    MatSortModule,\n    MatCheckbox,\n    TranslateModule,\n    QualityMatrixFilterComponent,\n    MatDateRangeInput,\n    MatDatepickerToggle,\n    MatDatepickerModule,\n    MatDateRangePicker,\n    MatDatepickerInput,\n    MatDatepicker,\n    DatepickerComponent,\n    ProgressSpinnerComponent,\n  ],\n  providers: [provideHttpClient(withInterceptorsFromDi()), provideCharts(withDefaultRegisterables())],\n})\nexport class NgMetaWidgetsLibModule {\n  constructor(private readonly svgIconsService: SvgIconsService) {\n    // we inject the service here to make sure the icons are registered (the init() method is called in the constructor)\n  }\n  static forRoot(params: NgMetaWidgetsLibConfiguration): ModuleWithProviders<NgMetaWidgetsLibModule> {\n    return {\n      ngModule: NgMetaWidgetsLibModule,\n      providers: [\n        ConfigHelperService,\n        { provide: 'config', useValue: params },\n        {\n          provide: JavaConfig,\n          useFactory: () => {\n            return new JavaConfig({\n              basePath: params.apiPath,\n              withCredentials: true,\n            });\n          },\n        },\n      ],\n    };\n  }\n}\n"]}
|