ngx-edu-sharing-metaqs2 0.9.37 → 0.9.38

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,40 +1,45 @@
1
- import { Component, computed, effect, Input, signal } from '@angular/core';
1
+ import { Component, computed, DestroyRef, Directive, inject, 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, zip } from 'rxjs';
5
+ import { BehaviorSubject, of, Subject } from 'rxjs';
6
6
  import { TranslateModule } from '@ngx-translate/core';
7
7
  import { AsyncPipe, NgForOf, NgIf } from '@angular/common';
8
+ import { MetaApiService } from '../meta-api.service';
8
9
  import { DateTime } from 'luxon';
9
- import { filter, finalize, map, skipWhile, switchMap, take, tap } from 'rxjs/operators';
10
+ import { distinctUntilChanged, filter, finalize, map, skipWhile, switchMap, take, tap, throttleTime, } from 'rxjs/operators';
10
11
  import { FormControl, FormGroup, FormsModule } from '@angular/forms';
11
12
  import { DatepickerComponent } from '../components/filter/datepicker/datepicker.component';
12
13
  import { MatIcon } from '@angular/material/icon';
13
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
14
+ import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
14
15
  import { MatSlideToggle } from '@angular/material/slide-toggle';
15
16
  import { ProgressSpinnerComponent } from '../components/loading_indicator/progress-spinner/progress-spinner.component';
17
+ import { EditorialLinkService } from '../components/editorial-link-service/editorial-link.service';
16
18
  import { MatRipple } from '@angular/material/core';
17
19
  import * as i0 from "@angular/core";
18
- import * as i1 from "../meta-api.service";
19
- import * as i2 from "../components/editorial-link-service/editorial-link.service";
20
- import * as i3 from "@angular/material/card";
21
- import * as i4 from "@angular/material/table";
22
- import * as i5 from "@ngx-translate/core";
23
- import * as i6 from "@angular/forms";
24
- export class CountsWithHistoryComponent {
25
- constructor(metaApi, destroyRef, linkService) {
26
- this.metaApi = metaApi;
27
- this.destroyRef = destroyRef;
28
- this.linkService = linkService;
20
+ import * as i1 from "@angular/material/card";
21
+ import * as i2 from "@angular/material/table";
22
+ import * as i3 from "@ngx-translate/core";
23
+ import * as i4 from "@angular/forms";
24
+ export class BaseHistoricDataTableDirective {
25
+ constructor() {
26
+ this.metaApi = inject(MetaApiService);
27
+ this.destroyRef = inject(DestroyRef);
28
+ this.linkService = inject(EditorialLinkService);
29
+ this.columnTranslationkey = null;
29
30
  this.loadingCount = signal(0);
30
31
  this.isLoading = computed(() => this.loadingCount() > 0);
32
+ this.isHistoryEnabled = signal(false);
33
+ this.isHistoryEnabled$ = toObservable(this.isHistoryEnabled);
31
34
  this.timeFilterLoaded = signal(false);
32
35
  this.recentTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
33
36
  this.pastTypeCount$ = new BehaviorSubject({ columns: [], rows: [] });
34
37
  this.DateTime = DateTime;
35
38
  this.columns = signal([]);
36
- this.apiMethod = 'getMaterialTypeCountsByReplicationSource';
37
- this.columnTranslationkey = null;
39
+ this.dataForPastData$ = new Subject();
40
+ this.dataForRecentData$ = new Subject();
41
+ this.startValues$ = this.dataForPastData$.pipe(filter((data) => this.validateLoadingData(data)), distinctUntilChanged((a, b) => this.compareLoadingDataEqual(a, b)), throttleTime(500), switchMap((start) => this.getCountByDate(start)), tap((it) => this.pastTypeCount$.next(it)), takeUntilDestroyed(this.destroyRef));
42
+ this.endValues$ = this.dataForRecentData$.pipe(filter((data) => this.validateLoadingData(data)), distinctUntilChanged((a, b) => this.compareLoadingDataEqual(a, b)), throttleTime(500), switchMap((end) => this.getCountByDate(end)), tap((it) => this.recentTypeCount$.next(it)), tap((it) => this.columns.set(it.columns.slice())), takeUntilDestroyed(this.destroyRef));
38
43
  /* In this widget's backend we do have data for today
39
44
  * therefore we set the end date to today and use the timerange() endpoint to only set the start date
40
45
  */
@@ -42,7 +47,11 @@ export class CountsWithHistoryComponent {
42
47
  start: new FormControl(),
43
48
  end: new FormControl(DateTime.utc().startOf('day'), { nonNullable: true }),
44
49
  });
45
- this.isHistoryEnabled = signal(false);
50
+ this.timerangeStart$ = of(undefined).pipe(switchMap(() => this.getAvailableDateRange()), take(1), map((rangeFilter) => this.getStartDateOfRange(rangeFilter)), tap((startDate) => {
51
+ //this is to have a default value for the start date => the min date of the range
52
+ this.range.setControl('start', new FormControl(startDate, { nonNullable: true }));
53
+ this.timeFilterLoaded.set(true);
54
+ }), takeUntilDestroyed(this.destroyRef));
46
55
  this.allColumns = computed(() => {
47
56
  if (!this.isHistoryEnabled()) {
48
57
  return this.recentColumns();
@@ -58,39 +67,9 @@ export class CountsWithHistoryComponent {
58
67
  this.pastColumns = computed(() => {
59
68
  return this.columns().map((c) => c.id + '_past');
60
69
  });
61
- effect(() => {
62
- this.range.controls.end.reset();
63
- if (!this.isHistoryEnabled()) {
64
- this.range.controls.start.setValue(this.range.controls.end.value.startOf('day'));
65
- }
66
- else {
67
- this.range.controls.start.reset();
68
- }
69
- });
70
70
  }
71
- ngOnInit() {
72
- this.getAvailableDateRange()
73
- .pipe(tap((rangeFilter) => {
74
- const startDate = this.getStartDateOfRange(rangeFilter);
75
- //this is to have a default value for the start date => the min date of the range
76
- this.range.setControl('start', new FormControl(startDate, { nonNullable: true }));
77
- }), finalize(() => {
78
- this.timeFilterLoaded.set(true);
79
- this.range.controls.start.reset();
80
- }), takeUntilDestroyed(this.destroyRef))
81
- .subscribe();
82
- this.getCountByDate(this.range.controls.end.value)
83
- .pipe(tap((response) => {
84
- this.columns.set(response.columns.slice());
85
- this.recentTypeCount$.next(response);
86
- }), takeUntilDestroyed(this.destroyRef))
87
- .subscribe();
88
- this.getValuesInDateRange()
89
- .pipe(tap(([past, recent]) => {
90
- this.pastTypeCount$.next(past);
91
- this.recentTypeCount$.next(recent);
92
- }), takeUntilDestroyed(this.destroyRef))
93
- .subscribe();
71
+ getAvailableDateRange() {
72
+ return of(undefined).pipe(tap(() => this.loadingCount.update((it) => it + 1)), switchMap(() => this.metaApi.getTimerangeFilter()), filter((filter) => filter != null), tap(() => this.loadingCount.update((it) => it - 1)));
94
73
  }
95
74
  getStartDateOfRange(rangeFilter) {
96
75
  return DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label, {
@@ -100,26 +79,6 @@ export class CountsWithHistoryComponent {
100
79
  columnIdent(_index, col) {
101
80
  return col.id;
102
81
  }
103
- getAvailableDateRange() {
104
- this.loadingCount.update((it) => it + 1);
105
- return this.metaApi.getTimerangeFilter().pipe(filter((filter) => filter != null), finalize(() => this.loadingCount.update((it) => it - 1)));
106
- }
107
- getValuesInDateRange() {
108
- return this.range.valueChanges.pipe(filter((range) => !!range.start?.isValid && !!range.end?.isValid), switchMap((range) => {
109
- return zip(this.getCountByDate(range.start), this.getCountByDate(range.end));
110
- }));
111
- }
112
- getCountByDate(date) {
113
- this.loadingCount.update((it) => it + 1);
114
- const filter = {
115
- field: 'asOf',
116
- values: [{ id: date.toISO({ includeOffset: false }), label: '' }],
117
- };
118
- return this.metaApi[this.apiMethod]([filter]).pipe(finalize(() => this.loadingCount.update((it) => it - 1)), map((response) => ({
119
- ...response,
120
- rows: response.rows.toSorted((a, b) => a.meta.label.localeCompare(b.meta.label)),
121
- })));
122
- }
123
82
  pastTypeCount(row, columnid) {
124
83
  if (!this.pastTypeCount$.value.rows.length) {
125
84
  return {};
@@ -144,8 +103,75 @@ export class CountsWithHistoryComponent {
144
103
  return this.linkService.createLinkForCountsWithHistory(this.sourceType, sourceId, issueId);
145
104
  }));
146
105
  }
147
- 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 }); }
148
- 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 <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker style=\"display: inline-block\" matRipple #datepickerRipple=\"matRipple\" [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\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 mat-cell-level-{{row.meta.level + 1}}\"\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}.toolbar,.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"] }, { kind: "directive", type: MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }] }); }
106
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BaseHistoricDataTableDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
107
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: BaseHistoricDataTableDirective, inputs: { apiMethod: "apiMethod", columnTranslationkey: "columnTranslationkey", pageTitle: "pageTitle", sourceType: "sourceType" }, ngImport: i0 }); }
108
+ }
109
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BaseHistoricDataTableDirective, decorators: [{
110
+ type: Directive
111
+ }], propDecorators: { apiMethod: [{
112
+ type: Input
113
+ }], columnTranslationkey: [{
114
+ type: Input
115
+ }], pageTitle: [{
116
+ type: Input
117
+ }], sourceType: [{
118
+ type: Input,
119
+ args: [{ required: true }]
120
+ }] } });
121
+ export class CountsWithHistoryComponent extends BaseHistoricDataTableDirective {
122
+ ngOnInit() {
123
+ this.apiMethod ??= 'getMaterialTypeCountsByReplicationSource';
124
+ this.range.valueChanges
125
+ .pipe(tap((data) => {
126
+ this.dataForPastData$.next({ date: data.start });
127
+ this.dataForRecentData$.next({ date: data.end });
128
+ }), takeUntilDestroyed(this.destroyRef))
129
+ .subscribe();
130
+ this.isHistoryEnabled$
131
+ .pipe(tap((enabled) => {
132
+ this.range.controls.end.reset();
133
+ if (!enabled) {
134
+ this.startDateSubscription?.unsubscribe();
135
+ this.startValuesSubscription?.unsubscribe();
136
+ return;
137
+ }
138
+ this.range.controls.start.reset();
139
+ if (!this.timeFilterLoaded()) {
140
+ this.startDateSubscription = this.timerangeStart$
141
+ .pipe(take(1), tap(() => {
142
+ this.startValuesSubscription = this.startValues$.subscribe();
143
+ this.range.controls.start.reset();
144
+ }))
145
+ .subscribe();
146
+ }
147
+ else {
148
+ this.startValuesSubscription = this.startValues$.subscribe();
149
+ this.range.controls.start.reset();
150
+ }
151
+ }), takeUntilDestroyed(this.destroyRef))
152
+ .subscribe();
153
+ this.endValues$.subscribe();
154
+ }
155
+ getCountByDate(data) {
156
+ const filter = {
157
+ field: 'asOf',
158
+ values: [{ id: data.date.toISO({ includeOffset: false }), label: '' }],
159
+ };
160
+ return of(undefined).pipe(tap(() => this.loadingCount.update((it) => it + 1)), switchMap(() => {
161
+ return this.metaApi[this.apiMethod]([filter]).pipe(finalize(() => this.loadingCount.update((it) => it - 1)));
162
+ }), map((response) => ({
163
+ ...response,
164
+ rows: response.rows.toSorted((a, b) => a.meta.label.localeCompare(b.meta.label)),
165
+ })));
166
+ }
167
+ validateLoadingData(data) {
168
+ return data.date?.isValid ?? false;
169
+ }
170
+ compareLoadingDataEqual(a, b) {
171
+ return a.date.equals(b.date);
172
+ }
173
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
174
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CountsWithHistoryComponent, isStandalone: true, selector: "metaqs2-counts-with-history", usesInheritance: true, 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 style=\"display: inline-block\" matRipple [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()\">\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 mat-cell-level-{{row.meta.level + 1}}\"\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}.toolbar,.actionbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}\n"], dependencies: [{ kind: "ngmodule", type: MatCardModule }, { kind: "component", type: i1.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "directive", type: i1.MatCardContent, selector: "mat-card-content" }, { kind: "component", type: i1.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i1.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: i2.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i2.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i2.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i2.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i2.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i2.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i2.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i2.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i2.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i2.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i3.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: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.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"] }, { kind: "directive", type: MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }] }); }
149
175
  }
150
176
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountsWithHistoryComponent, decorators: [{
151
177
  type: Component,
@@ -172,16 +198,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
172
198
  AsyncPipe,
173
199
  ProgressSpinnerComponent,
174
200
  MatRipple,
175
- ], 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 <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker style=\"display: inline-block\" matRipple #datepickerRipple=\"matRipple\" [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\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 mat-cell-level-{{row.meta.level + 1}}\"\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}.toolbar,.actionbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}\n"] }]
176
- }], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i0.DestroyRef }, { type: i2.EditorialLinkService }], propDecorators: { apiMethod: [{
177
- type: Input,
178
- args: [{ required: true }]
179
- }], columnTranslationkey: [{
180
- type: Input
181
- }], pageTitle: [{
182
- type: Input
183
- }], sourceType: [{
184
- type: Input,
185
- args: [{ required: true }]
186
- }] } });
187
- //# 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';\nimport { MatRipple } from '@angular/material/core';\n\nexport type MethodType =\n  | ((it: Filter[]) => Observable<MatrixWithCounts>)\n  | ((x: string, it: Filter[]) => Observable<MatrixWithCounts>);\n\nexport type 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    MatRipple,\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, MethodType> = 'getMaterialTypeCountsByReplicationSource';\n\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  protected 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  protected 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\n    return (this.metaApi[this.apiMethod] as (filters: Filter[]) => Observable<MatrixWithCounts>)([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    <div style=\"flex: 1 1 auto\"></div>\n    <metaqs2-datepicker style=\"display: inline-block\" matRipple #datepickerRipple=\"matRipple\" [disabled]=\"isLoading() || !isHistoryEnabled()\" [inputGroup]=\"range\" *ngIf=\"timeFilterLoaded() && isHistoryEnabled()\"></metaqs2-datepicker>\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 mat-cell-level-{{row.meta.level + 1}}\"\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"]}
201
+ ], 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 style=\"display: inline-block\" matRipple [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()\">\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 mat-cell-level-{{row.meta.level + 1}}\"\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}.toolbar,.actionbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}\n"] }]
202
+ }] });
203
+ //# 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, Directive, inject, 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, of, Subject, Unsubscribable } 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 {\n  distinctUntilChanged,\n  filter,\n  finalize,\n  map,\n  skipWhile,\n  switchMap,\n  take,\n  tap,\n  throttleTime,\n} 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, toObservable } 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';\nimport { MatRipple } from '@angular/material/core';\n\nexport type PickKeysByPropertyType<TObject extends object, TPropertyType> = {\n  [TKey in keyof TObject]: TObject[TKey] extends TPropertyType ? TKey : never;\n}[keyof TObject];\n\n@Directive()\nexport abstract class BaseHistoricDataTableDirective<\n  TLoadingData extends object,\n  TMethodType extends MetaApiService[keyof MetaApiService] & Function\n> {\n  protected readonly metaApi = inject(MetaApiService);\n  protected readonly destroyRef = inject(DestroyRef);\n  protected readonly linkService = inject(EditorialLinkService);\n\n  @Input()\n  apiMethod: PickKeysByPropertyType<MetaApiService, TMethodType>;\n  @Input()\n  columnTranslationkey: string | null = null;\n  @Input()\n  pageTitle: string;\n  @Input({ required: true })\n  sourceType: 'replicationSource' | 'collection';\n\n  protected loadingCount = signal(0);\n  protected readonly isLoading = computed(() => this.loadingCount() > 0);\n  readonly isHistoryEnabled = signal<boolean>(false);\n  protected readonly isHistoryEnabled$ = toObservable(this.isHistoryEnabled);\n\n  protected readonly timeFilterLoaded = signal(false);\n  protected recentTypeCount$ = new BehaviorSubject({ columns: [], rows: [] } as MatrixWithCounts);\n  protected pastTypeCount$ = new BehaviorSubject({ columns: [], rows: [] } as MatrixWithCounts);\n  protected readonly DateTime = DateTime;\n  public columns = signal([] as Array<QualityMatrixHeader>);\n\n  protected readonly dataForPastData$ = new Subject<Partial<TLoadingData>>();\n  protected readonly dataForRecentData$ = new Subject<Partial<TLoadingData>>();\n\n  protected readonly startValues$ = this.dataForPastData$.pipe(\n    filter((data) => this.validateLoadingData(data)),\n    distinctUntilChanged((a, b) => this.compareLoadingDataEqual(a, b)),\n    throttleTime(500),\n    switchMap((start) => this.getCountByDate(start)),\n    tap((it) => this.pastTypeCount$.next(it)),\n    takeUntilDestroyed(this.destroyRef)\n  );\n  protected readonly endValues$ = this.dataForRecentData$.pipe(\n    filter((data) => this.validateLoadingData(data)),\n    distinctUntilChanged((a, b) => this.compareLoadingDataEqual(a, b)),\n    throttleTime(500),\n    switchMap((end) => this.getCountByDate(end)),\n    tap((it) => this.recentTypeCount$.next(it)),\n    tap((it) => this.columns.set(it.columns.slice())),\n    takeUntilDestroyed(this.destroyRef)\n  );\n\n  protected startDateSubscription: Unsubscribable | undefined;\n  protected startValuesSubscription: Unsubscribable | undefined;\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  protected readonly timerangeStart$ = of(undefined).pipe(\n    switchMap(() => this.getAvailableDateRange()),\n    take(1),\n    map((rangeFilter) => this.getStartDateOfRange(rangeFilter)),\n    tap((startDate) => {\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      this.timeFilterLoaded.set(true);\n    }),\n    takeUntilDestroyed(this.destroyRef)\n  );\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  protected abstract getCountByDate(date: TLoadingData): Observable<MatrixWithCounts>;\n  protected abstract validateLoadingData(data: Partial<TLoadingData>): data is TLoadingData;\n  protected abstract compareLoadingDataEqual(a: TLoadingData, b: TLoadingData): boolean;\n\n  protected getAvailableDateRange() {\n    return of(undefined).pipe(\n      tap(() => this.loadingCount.update((it) => it + 1)),\n      switchMap(() => this.metaApi.getTimerangeFilter()),\n      filter((filter): filter is Filter => filter != null),\n      tap(() => this.loadingCount.update((it) => it - 1))\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  public columnIdent(_index: number, col: QualityMatrixHeader) {\n    return col.id;\n  }\n\n  public pastTypeCount(row: MatrixRowWithCounts, columnid: string): { 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  public 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\ntype LoadingData = {\n  date: DateTime;\n};\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    MatRipple,\n  ],\n  templateUrl: './counts-with-history.component.html',\n  styleUrl: './counts-with-history.component.scss',\n})\nexport class CountsWithHistoryComponent\n  extends BaseHistoricDataTableDirective<LoadingData, (filters: Filter[]) => Observable<MatrixWithCounts>>\n  implements OnInit\n{\n  ngOnInit(): void {\n    this.apiMethod ??= 'getMaterialTypeCountsByReplicationSource';\n\n    this.range.valueChanges\n      .pipe(\n        tap((data) => {\n          this.dataForPastData$.next({ date: data.start });\n          this.dataForRecentData$.next({ date: data.end });\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    this.isHistoryEnabled$\n      .pipe(\n        tap((enabled) => {\n          this.range.controls.end.reset();\n          if (!enabled) {\n            this.startDateSubscription?.unsubscribe();\n            this.startValuesSubscription?.unsubscribe();\n            return;\n          }\n          this.range.controls.start.reset();\n          if (!this.timeFilterLoaded()) {\n            this.startDateSubscription = this.timerangeStart$\n              .pipe(\n                take(1),\n                tap(() => {\n                  this.startValuesSubscription = this.startValues$.subscribe();\n                  this.range.controls.start.reset();\n                })\n              )\n              .subscribe();\n          } else {\n            this.startValuesSubscription = this.startValues$.subscribe();\n            this.range.controls.start.reset();\n          }\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    this.endValues$.subscribe();\n  }\n\n  protected getCountByDate(data: LoadingData): Observable<MatrixWithCounts> {\n    const filter: Filter = {\n      field: 'asOf',\n      values: [{ id: data.date.toISO({ includeOffset: false })!, label: '' }],\n    };\n\n    return of(undefined).pipe(\n      tap(() => this.loadingCount.update((it) => it + 1)),\n      switchMap(() => {\n        return this.metaApi[this.apiMethod]([filter]).pipe(finalize(() => this.loadingCount.update((it) => it - 1)));\n      }),\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 validateLoadingData(data: Partial<LoadingData>): data is LoadingData {\n    return data.date?.isValid ?? false;\n  }\n\n  protected compareLoadingDataEqual(a: LoadingData, b: LoadingData): boolean {\n    return a.date.equals(b.date);\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 style=\"display: inline-block\" matRipple [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()\">\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 mat-cell-level-{{row.meta.level + 1}}\"\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"]}
@@ -74,4 +74,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
74
74
  providedIn: 'root',
75
75
  }]
76
76
  }], ctorParameters: () => [{ type: i1.CollectionAPIService }, { type: i1.ReplicationSourceAPIService }, { type: i1.FilterAPIService }, { type: i1.EditorsAPIService }] });
77
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0YS1hcGkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL25nLW1ldGEtd2lkZ2V0cy1saWIvc3JjL2xpYi9tZXRhLWFwaS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFjM0MsT0FBTyxFQUFFLEdBQUcsRUFBRSxJQUFJLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7O0FBSzNDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGNBQWM7SUFDekIsWUFDVSxjQUFvQyxFQUNwQyxlQUE0QyxFQUM1QyxTQUEyQixFQUMzQixVQUE2QjtRQUg3QixtQkFBYyxHQUFkLGNBQWMsQ0FBc0I7UUFDcEMsb0JBQWUsR0FBZixlQUFlLENBQTZCO1FBQzVDLGNBQVMsR0FBVCxTQUFTLENBQWtCO1FBQzNCLGVBQVUsR0FBVixVQUFVLENBQW1CO0lBQ3BDLENBQUM7SUFFSjs7O09BR0c7SUFDSSxnQkFBZ0I7UUFDckIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLHVCQUF1QixFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2hFLENBQUM7SUFDTSxrQkFBa0I7UUFDdkIsT0FBTyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsS0FBSyxLQUFLLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsSCxDQUFDO0lBQ00sb0JBQW9CO1FBQ3pCLE9BQU8sSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO0lBQ3JELENBQUM7SUFDRDs7O09BR0c7SUFDSSxrQkFBa0I7UUFDdkIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLHNCQUFzQixFQUFFLENBQUMsSUFBSSxDQUNqRCxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQ1AsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FDN0IsQ0FBQztJQUNKLENBQUM7SUFFRCxtQ0FBbUMsQ0FBQyxPQUFlLEVBQUUsT0FBaUI7UUFDcEUsT0FBTyxJQUFJLENBQUMsY0FBYyxDQUFDLG1DQUFtQyxDQUFDLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQztJQUNuRixDQUFDO0lBRUQseUJBQXlCLENBQUMsS0FBYTtRQUNyQyxPQUFPLElBQUksQ0FBQyxjQUFjLENBQUMsZUFBZSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3BELENBQUM7SUFFRCw2QkFBNkIsQ0FBQyxJQUFtQjtRQUMvQyxPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDdkQsQ0FBQztJQUVELDBCQUEwQixDQUFDLElBQXlCO1FBQ2xELE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNoRCxDQUFDO0lBRUQsd0NBQXdDLENBQUMsSUFBYztRQUNyRCxPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsZ0RBQWdELENBQUMsSUFBSSxDQUFDLENBQUM7SUFDckYsQ0FBQztJQUVELG1DQUFtQyxDQUFDLElBQWM7UUFDaEQsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLG1EQUFtRCxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3hGLENBQUM7SUFFRCw4QkFBOEIsQ0FBQyxLQUFhLEVBQUUsT0FBaUI7UUFDN0QsT0FBTyxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxDQUM5RCxLQUFLLEVBQ0wsT0FBTyxDQUNtQyxDQUFDO0lBQy9DLENBQUM7SUFFRCxvQ0FBb0MsQ0FBQyxJQUFjO1FBQ2pELE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxvQ0FBb0MsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNwRSxDQUFDO0lBRUQscUNBQXFDLENBQUMsSUFBYztRQUNsRCxPQUFPLElBQUksQ0FBQyxVQUFVLENBQUMscUNBQXFDLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDckUsQ0FBQztJQUVELG9DQUFvQyxDQUFDLElBQWM7UUFDakQsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLG9DQUFvQyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3pFLENBQUM7SUFFRCx1QkFBdUI7UUFDckIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLHVCQUF1QixFQUFFLENBQUM7SUFDbEQsQ0FBQzsrR0E3RVUsY0FBYzttSEFBZCxjQUFjLGNBTGIsTUFBTTs7NEZBS1AsY0FBYztrQkFOMUIsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5cbmltcG9ydCB7XG4gIENvbGxlY3Rpb25BUElTZXJ2aWNlLFxuICBFZGl0b3JzQVBJU2VydmljZSxcbiAgRWR1Q29sbGVjdGlvbixcbiAgRmlsdGVyLFxuICBGaWx0ZXJBUElTZXJ2aWNlLFxuICBNYXRlcmlhbENvdW50RmlsdGVyLFxuICBNYXRyaXhXaXRoQ291bnRzLFxuICBRdWFsaXR5TWF0cml4LFxuICBSZXBsaWNhdGlvblNvdXJjZUFQSVNlcnZpY2UsXG59IGZyb20gJy4vamF2YS1hcGknO1xuaW1wb3J0IHsgbWFwLCB0YWtlIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG4vKipcbiAqIFRoaXMgY2xhc3MgaXMgYSB0aGluIHdyYXBwZXIgYXJvdW5kIHRoZSBBUEkgc2VydmljZXMuXG4gKi9cbmV4cG9ydCBjbGFzcyBNZXRhQXBpU2VydmljZSB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgY29sbGVjdGlvbnNBUEk6IENvbGxlY3Rpb25BUElTZXJ2aWNlLFxuICAgIHByaXZhdGUgcmVwbGljYXRpb25zQVBJOiBSZXBsaWNhdGlvblNvdXJjZUFQSVNlcnZpY2UsXG4gICAgcHJpdmF0ZSBmaWx0ZXJBUEk6IEZpbHRlckFQSVNlcnZpY2UsXG4gICAgcHJpdmF0ZSBlZGl0b3JzQVBJOiBFZGl0b3JzQVBJU2VydmljZVxuICApIHt9XG5cbiAgLyoqXG4gICAqIEdldCB0aGUgZmlsdGVycyBmb3IgdGhlIHF1YWxpdHkgbWF0cml4LlxuICAgKiBJdCBwaXBlcyB0aGUgb2JzZXJ2YWJsZSB0byB0YWtlIG9ubHkgb25lIHZhbHVlLlxuICAgKi9cbiAgcHVibGljIGdldFNlYXJjaEZpbHRlcnMoKTogT2JzZXJ2YWJsZTxGaWx0ZXJbXT4ge1xuICAgIHJldHVybiB0aGlzLmZpbHRlckFQSS5nZXRRdWFsaXR4TWF0cml4RmlsdGVycygpLnBpcGUodGFrZSgxKSk7XG4gIH1cbiAgcHVibGljIGdldENhdGVnb3J5RmlsdGVycygpOiBPYnNlcnZhYmxlPEZpbHRlcltdPiB7XG4gICAgcmV0dXJuIHRoaXMuZ2V0U2VhcmNoRmlsdGVycygpLnBpcGUobWFwKChmaWx0ZXJzKSA9PiBmaWx0ZXJzLmZpbHRlcigoZmlsdGVyKSA9PiBmaWx0ZXIuZmllbGQgIT09ICd0aW1lcmFuZ2UnKSkpO1xuICB9XG4gIHB1YmxpYyBnZXRDb2xsZWN0aW9uc0ZpbHRlcigpOiBPYnNlcnZhYmxlPEFycmF5PEVkdUNvbGxlY3Rpb24+PiB7XG4gICAgcmV0dXJuIHRoaXMuY29sbGVjdGlvbnNBUEkuZ2V0VG9wTGV2ZWxDb2xsZWN0aW9uKCk7XG4gIH1cbiAgLyoqXG4gICAqIEdldCB0aGUgdGltZXJhbmdlIGZpbHRlciBmb3IgdGhlIHF1YWxpdHkgbWF0cml4LlxuICAgKiBSZXR1cm5zIHRoZSBmaXJzdCBmaWx0ZXIgb2YgdGhlIGhpc3RvcmljYWwgdGltZXJhbmdlIGZpbHRlcnMuXG4gICAqL1xuICBwdWJsaWMgZ2V0VGltZXJhbmdlRmlsdGVyKCk6IE9ic2VydmFibGU8RmlsdGVyIHwgdW5kZWZpbmVkPiB7XG4gICAgcmV0dXJuIHRoaXMuZmlsdGVyQVBJLmdldEhpc3RvcmljYWxUaW1lcmFuZ2UoKS5waXBlKFxuICAgICAgdGFrZSgxKSxcbiAgICAgIG1hcCgoZmlsdGVycykgPT4gZmlsdGVyc1swXSlcbiAgICApO1xuICB9XG5cbiAgZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeFBlckNvbGxlY3Rpb24obm9kZVJlZjogc3RyaW5nLCBvZXJPbmx5PzogYm9vbGVhbik6IE9ic2VydmFibGU8UXVhbGl0eU1hdHJpeD4ge1xuICAgIHJldHVybiB0aGlzLmNvbGxlY3Rpb25zQVBJLmdldE1hdGVyaWFsQ291bnRNYXRyaXhQZXJDb2xsZWN0aW9uKG5vZGVSZWYsIG9lck9ubHkpO1xuICB9XG5cbiAgZ2V0Q29sbGVjdGlvbkNvbXBsZXRlbmVzcyhjb2xJZDogc3RyaW5nKTogT2JzZXJ2YWJsZTxNYXRyaXhXaXRoQ291bnRzPiB7XG4gICAgcmV0dXJuIHRoaXMuY29sbGVjdGlvbnNBUEkuZ2V0Q29tcGxldGVuZXNzKGNvbElkKTtcbiAgfVxuXG4gIGdldFF1YWxpdHlNYXRyaXhXaXRoRmlsdGVyc1YyKGJvZHk6IEFycmF5PEZpbHRlcj4pIHtcbiAgICByZXR1cm4gdGhpcy5yZXBsaWNhdGlvbnNBUEkuZ2V0UXVhbGl0eU1hdHJpeFYyKGJvZHkpO1xuICB9XG5cbiAgZ2V0RWRpdG9yaWFsTWF0ZXJpYWxDb3VudHMoYm9keTogTWF0ZXJpYWxDb3VudEZpbHRlcikge1xuICAgIHJldHVybiB0aGlzLmVkaXRvcnNBUEkuZ2V0TWF0ZXJpYWxDb3VudChib2R5KTtcbiAgfVxuXG4gIGdldE1hdGVyaWFsVHlwZUNvdW50c0J5UmVwbGljYXRpb25Tb3VyY2UoYm9keTogRmlsdGVyW10pIHtcbiAgICByZXR1cm4gdGhpcy5yZXBsaWNhdGlvbnNBUEkuZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeEJ5UmVwbGljYXRpb25Tb3VyY2VBbmRUeXBlKGJvZHkpO1xuICB9XG5cbiAgZ2V0TGljZW5zZUNvdW50c0J5UmVwbGljYXRpb25Tb3VyY2UoYm9keTogRmlsdGVyW10pIHtcbiAgICByZXR1cm4gdGhpcy5yZXBsaWNhdGlvbnNBUEkuZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeEJ5UmVwbGljYXRpb25Tb3VyY2VBbmRMaWNlbnNlKGJvZHkpO1xuICB9XG5cbiAgZ2V0TGljZW5zZUNvdW50c0J5TGljZW5zZUdyb3VwKGNvbElkOiBzdHJpbmcsIGZpbHRlcnM6IEZpbHRlcltdKTogT2JzZXJ2YWJsZTxNYXRyaXhXaXRoQ291bnRzPiB7XG4gICAgcmV0dXJuIHRoaXMuY29sbGVjdGlvbnNBUEkuZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeEJ5TGljZW5zZUdyb3VwMShcbiAgICAgIGNvbElkLFxuICAgICAgZmlsdGVyc1xuICAgICkgYXMgdW5rbm93biBhcyBPYnNlcnZhYmxlPE1hdHJpeFdpdGhDb3VudHM+O1xuICB9XG5cbiAgZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeEJ5TGljZW5zZUdyb3VwKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMuZWRpdG9yc0FQSS5nZXRNYXRlcmlhbENvdW50TWF0cml4QnlMaWNlbnNlR3JvdXAoYm9keSk7XG4gIH1cblxuICBnZXRDb21wbGV0ZW5lc3NGb3JEaXNjaXBsaW5hcnlQb3J0YWxzKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMuZWRpdG9yc0FQSS5nZXRDb21wbGV0ZW5lc3NGb3JEaXNjaXBsaW5hcnlQb3J0YWxzKGJvZHkpO1xuICB9XG5cbiAgZ2V0Q29tcGxldGVuZXNzRm9yUmVwbGljYXRpb25Tb3VyY2VzKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMucmVwbGljYXRpb25zQVBJLmdldENvbXBsZXRlbmVzc0ZvclJlcGxpY2F0aW9uU291cmNlcyhib2R5KTtcbiAgfVxuXG4gIGdldE1hdGVyaWFsVHlwZXNNYXBwaW5nKCkge1xuICAgIHJldHVybiB0aGlzLmZpbHRlckFQSS5nZXRNYXRlcmlhbFR5cGVzTWFwcGluZygpO1xuICB9XG59XG4iXX0=
77
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0YS1hcGkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL25nLW1ldGEtd2lkZ2V0cy1saWIvc3JjL2xpYi9tZXRhLWFwaS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFjM0MsT0FBTyxFQUFFLEdBQUcsRUFBRSxJQUFJLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7O0FBSzNDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGNBQWM7SUFDekIsWUFDVSxjQUFvQyxFQUNwQyxlQUE0QyxFQUM1QyxTQUEyQixFQUMzQixVQUE2QjtRQUg3QixtQkFBYyxHQUFkLGNBQWMsQ0FBc0I7UUFDcEMsb0JBQWUsR0FBZixlQUFlLENBQTZCO1FBQzVDLGNBQVMsR0FBVCxTQUFTLENBQWtCO1FBQzNCLGVBQVUsR0FBVixVQUFVLENBQW1CO0lBQ3BDLENBQUM7SUFFSjs7O09BR0c7SUFDSSxnQkFBZ0I7UUFDckIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLHVCQUF1QixFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2hFLENBQUM7SUFDTSxrQkFBa0I7UUFDdkIsT0FBTyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsS0FBSyxLQUFLLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsSCxDQUFDO0lBQ00sb0JBQW9CO1FBQ3pCLE9BQU8sSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO0lBQ3JELENBQUM7SUFDRDs7O09BR0c7SUFDSSxrQkFBa0I7UUFDdkIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLHNCQUFzQixFQUFFLENBQUMsSUFBSSxDQUNqRCxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQ1AsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FDN0IsQ0FBQztJQUNKLENBQUM7SUFFRCxtQ0FBbUMsQ0FBQyxPQUFlLEVBQUUsT0FBaUI7UUFDcEUsT0FBTyxJQUFJLENBQUMsY0FBYyxDQUFDLG1DQUFtQyxDQUFDLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQztJQUNuRixDQUFDO0lBRUQseUJBQXlCLENBQUMsS0FBYTtRQUNyQyxPQUFPLElBQUksQ0FBQyxjQUFjLENBQUMsZUFBZSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3BELENBQUM7SUFFRCw2QkFBNkIsQ0FBQyxJQUFtQjtRQUMvQyxPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDdkQsQ0FBQztJQUVELDBCQUEwQixDQUFDLElBQXlCO1FBQ2xELE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNoRCxDQUFDO0lBRUQsd0NBQXdDLENBQUMsSUFBYztRQUNyRCxPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsZ0RBQWdELENBQUMsSUFBSSxDQUFDLENBQUM7SUFDckYsQ0FBQztJQUVELG1DQUFtQyxDQUFDLElBQWM7UUFDaEQsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLG1EQUFtRCxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3hGLENBQUM7SUFFRCw4QkFBOEIsQ0FBQyxLQUFhLEVBQUUsT0FBaUI7UUFDN0QsT0FBTyxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxDQUFDLEtBQUssRUFBRSxPQUFPLENBQUMsQ0FBQztJQUNuRixDQUFDO0lBRUQsb0NBQW9DLENBQUMsSUFBYztRQUNqRCxPQUFPLElBQUksQ0FBQyxVQUFVLENBQUMsb0NBQW9DLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDcEUsQ0FBQztJQUVELHFDQUFxQyxDQUFDLElBQWM7UUFDbEQsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLHFDQUFxQyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3JFLENBQUM7SUFFRCxvQ0FBb0MsQ0FBQyxJQUFjO1FBQ2pELE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxvQ0FBb0MsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUN6RSxDQUFDO0lBRUQsdUJBQXVCO1FBQ3JCLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyx1QkFBdUIsRUFBRSxDQUFDO0lBQ2xELENBQUM7K0dBMUVVLGNBQWM7bUhBQWQsY0FBYyxjQUxiLE1BQU07OzRGQUtQLGNBQWM7a0JBTjFCLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSB9IGZyb20gJ3J4anMnO1xuXG5pbXBvcnQge1xuICBDb2xsZWN0aW9uQVBJU2VydmljZSxcbiAgRWRpdG9yc0FQSVNlcnZpY2UsXG4gIEVkdUNvbGxlY3Rpb24sXG4gIEZpbHRlcixcbiAgRmlsdGVyQVBJU2VydmljZSxcbiAgTWF0ZXJpYWxDb3VudEZpbHRlcixcbiAgTWF0cml4V2l0aENvdW50cyxcbiAgUXVhbGl0eU1hdHJpeCxcbiAgUmVwbGljYXRpb25Tb3VyY2VBUElTZXJ2aWNlLFxufSBmcm9tICcuL2phdmEtYXBpJztcbmltcG9ydCB7IG1hcCwgdGFrZSB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuLyoqXG4gKiBUaGlzIGNsYXNzIGlzIGEgdGhpbiB3cmFwcGVyIGFyb3VuZCB0aGUgQVBJIHNlcnZpY2VzLlxuICovXG5leHBvcnQgY2xhc3MgTWV0YUFwaVNlcnZpY2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGNvbGxlY3Rpb25zQVBJOiBDb2xsZWN0aW9uQVBJU2VydmljZSxcbiAgICBwcml2YXRlIHJlcGxpY2F0aW9uc0FQSTogUmVwbGljYXRpb25Tb3VyY2VBUElTZXJ2aWNlLFxuICAgIHByaXZhdGUgZmlsdGVyQVBJOiBGaWx0ZXJBUElTZXJ2aWNlLFxuICAgIHByaXZhdGUgZWRpdG9yc0FQSTogRWRpdG9yc0FQSVNlcnZpY2VcbiAgKSB7fVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIGZpbHRlcnMgZm9yIHRoZSBxdWFsaXR5IG1hdHJpeC5cbiAgICogSXQgcGlwZXMgdGhlIG9ic2VydmFibGUgdG8gdGFrZSBvbmx5IG9uZSB2YWx1ZS5cbiAgICovXG4gIHB1YmxpYyBnZXRTZWFyY2hGaWx0ZXJzKCk6IE9ic2VydmFibGU8RmlsdGVyW10+IHtcbiAgICByZXR1cm4gdGhpcy5maWx0ZXJBUEkuZ2V0UXVhbGl0eE1hdHJpeEZpbHRlcnMoKS5waXBlKHRha2UoMSkpO1xuICB9XG4gIHB1YmxpYyBnZXRDYXRlZ29yeUZpbHRlcnMoKTogT2JzZXJ2YWJsZTxGaWx0ZXJbXT4ge1xuICAgIHJldHVybiB0aGlzLmdldFNlYXJjaEZpbHRlcnMoKS5waXBlKG1hcCgoZmlsdGVycykgPT4gZmlsdGVycy5maWx0ZXIoKGZpbHRlcikgPT4gZmlsdGVyLmZpZWxkICE9PSAndGltZXJhbmdlJykpKTtcbiAgfVxuICBwdWJsaWMgZ2V0Q29sbGVjdGlvbnNGaWx0ZXIoKTogT2JzZXJ2YWJsZTxBcnJheTxFZHVDb2xsZWN0aW9uPj4ge1xuICAgIHJldHVybiB0aGlzLmNvbGxlY3Rpb25zQVBJLmdldFRvcExldmVsQ29sbGVjdGlvbigpO1xuICB9XG4gIC8qKlxuICAgKiBHZXQgdGhlIHRpbWVyYW5nZSBmaWx0ZXIgZm9yIHRoZSBxdWFsaXR5IG1hdHJpeC5cbiAgICogUmV0dXJucyB0aGUgZmlyc3QgZmlsdGVyIG9mIHRoZSBoaXN0b3JpY2FsIHRpbWVyYW5nZSBmaWx0ZXJzLlxuICAgKi9cbiAgcHVibGljIGdldFRpbWVyYW5nZUZpbHRlcigpOiBPYnNlcnZhYmxlPEZpbHRlciB8IHVuZGVmaW5lZD4ge1xuICAgIHJldHVybiB0aGlzLmZpbHRlckFQSS5nZXRIaXN0b3JpY2FsVGltZXJhbmdlKCkucGlwZShcbiAgICAgIHRha2UoMSksXG4gICAgICBtYXAoKGZpbHRlcnMpID0+IGZpbHRlcnNbMF0pXG4gICAgKTtcbiAgfVxuXG4gIGdldE1hdGVyaWFsQ291bnRNYXRyaXhQZXJDb2xsZWN0aW9uKG5vZGVSZWY6IHN0cmluZywgb2VyT25seT86IGJvb2xlYW4pOiBPYnNlcnZhYmxlPFF1YWxpdHlNYXRyaXg+IHtcbiAgICByZXR1cm4gdGhpcy5jb2xsZWN0aW9uc0FQSS5nZXRNYXRlcmlhbENvdW50TWF0cml4UGVyQ29sbGVjdGlvbihub2RlUmVmLCBvZXJPbmx5KTtcbiAgfVxuXG4gIGdldENvbGxlY3Rpb25Db21wbGV0ZW5lc3MoY29sSWQ6IHN0cmluZyk6IE9ic2VydmFibGU8TWF0cml4V2l0aENvdW50cz4ge1xuICAgIHJldHVybiB0aGlzLmNvbGxlY3Rpb25zQVBJLmdldENvbXBsZXRlbmVzcyhjb2xJZCk7XG4gIH1cblxuICBnZXRRdWFsaXR5TWF0cml4V2l0aEZpbHRlcnNWMihib2R5OiBBcnJheTxGaWx0ZXI+KSB7XG4gICAgcmV0dXJuIHRoaXMucmVwbGljYXRpb25zQVBJLmdldFF1YWxpdHlNYXRyaXhWMihib2R5KTtcbiAgfVxuXG4gIGdldEVkaXRvcmlhbE1hdGVyaWFsQ291bnRzKGJvZHk6IE1hdGVyaWFsQ291bnRGaWx0ZXIpIHtcbiAgICByZXR1cm4gdGhpcy5lZGl0b3JzQVBJLmdldE1hdGVyaWFsQ291bnQoYm9keSk7XG4gIH1cblxuICBnZXRNYXRlcmlhbFR5cGVDb3VudHNCeVJlcGxpY2F0aW9uU291cmNlKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMucmVwbGljYXRpb25zQVBJLmdldE1hdGVyaWFsQ291bnRNYXRyaXhCeVJlcGxpY2F0aW9uU291cmNlQW5kVHlwZShib2R5KTtcbiAgfVxuXG4gIGdldExpY2Vuc2VDb3VudHNCeVJlcGxpY2F0aW9uU291cmNlKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMucmVwbGljYXRpb25zQVBJLmdldE1hdGVyaWFsQ291bnRNYXRyaXhCeVJlcGxpY2F0aW9uU291cmNlQW5kTGljZW5zZShib2R5KTtcbiAgfVxuXG4gIGdldExpY2Vuc2VDb3VudHNCeUxpY2Vuc2VHcm91cChjb2xJZDogc3RyaW5nLCBmaWx0ZXJzOiBGaWx0ZXJbXSk6IE9ic2VydmFibGU8UXVhbGl0eU1hdHJpeD4ge1xuICAgIHJldHVybiB0aGlzLmNvbGxlY3Rpb25zQVBJLmdldE1hdGVyaWFsQ291bnRNYXRyaXhCeUxpY2Vuc2VHcm91cDEoY29sSWQsIGZpbHRlcnMpO1xuICB9XG5cbiAgZ2V0TWF0ZXJpYWxDb3VudE1hdHJpeEJ5TGljZW5zZUdyb3VwKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMuZWRpdG9yc0FQSS5nZXRNYXRlcmlhbENvdW50TWF0cml4QnlMaWNlbnNlR3JvdXAoYm9keSk7XG4gIH1cblxuICBnZXRDb21wbGV0ZW5lc3NGb3JEaXNjaXBsaW5hcnlQb3J0YWxzKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMuZWRpdG9yc0FQSS5nZXRDb21wbGV0ZW5lc3NGb3JEaXNjaXBsaW5hcnlQb3J0YWxzKGJvZHkpO1xuICB9XG5cbiAgZ2V0Q29tcGxldGVuZXNzRm9yUmVwbGljYXRpb25Tb3VyY2VzKGJvZHk6IEZpbHRlcltdKSB7XG4gICAgcmV0dXJuIHRoaXMucmVwbGljYXRpb25zQVBJLmdldENvbXBsZXRlbmVzc0ZvclJlcGxpY2F0aW9uU291cmNlcyhib2R5KTtcbiAgfVxuXG4gIGdldE1hdGVyaWFsVHlwZXNNYXBwaW5nKCkge1xuICAgIHJldHVybiB0aGlzLmZpbHRlckFQSS5nZXRNYXRlcmlhbFR5cGVzTWFwcGluZygpO1xuICB9XG59XG4iXX0=