ngx-edu-sharing-metaqs2 0.9.53 → 0.9.54

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (76) hide show
  1. package/README.md +1 -14
  2. package/fesm2022/ngx-edu-sharing-metaqs2.mjs +195 -197
  3. package/fesm2022/ngx-edu-sharing-metaqs2.mjs.map +1 -1
  4. package/lib/components/collection-issues/collection-issues.component.d.ts +1 -1
  5. package/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.d.ts +1 -1
  6. package/lib/components/filter/quality-matrix-filter.component.d.ts +4 -6
  7. package/lib/components/material-issues/material-issues.component.d.ts +1 -1
  8. package/lib/components/node-entry/node-entry.component.d.ts +1 -1
  9. package/lib/components/node-list/node-list.component.d.ts +1 -1
  10. package/lib/components/quality-matrix/quality_matrix.d.ts +1 -3
  11. package/lib/counts-with-history/counts-with-history.component.d.ts +1 -1
  12. package/lib/ng-meta-widgets-lib.module.d.ts +34 -37
  13. package/lib/node-image-url.pipe.d.ts +1 -1
  14. package/lib/wrap-observable.pipe.d.ts +1 -1
  15. package/package.json +6 -8
  16. package/esm2022/lib/collection-count-history/collection-count-history.component.mjs +0 -131
  17. package/esm2022/lib/collection-count-history/monthpicker/monthpicker.component.mjs +0 -114
  18. package/esm2022/lib/components/collection-issues/collection-issues.component.mjs +0 -23
  19. package/esm2022/lib/components/donut-chart/donut-chart.component.mjs +0 -85
  20. package/esm2022/lib/components/donut-chart/donut-chart.model.mjs +0 -2
  21. package/esm2022/lib/components/donut-chart/donut-chart.pipe.mjs +0 -50
  22. package/esm2022/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.mjs +0 -79
  23. package/esm2022/lib/components/editorial-link-service/editorial-link.service.mjs +0 -169
  24. package/esm2022/lib/components/filter/datepicker/datepicker.component.mjs +0 -99
  25. package/esm2022/lib/components/filter/quality-matrix-filter.component.mjs +0 -47
  26. package/esm2022/lib/components/loading_indicator/overlay/overlay.service.mjs +0 -41
  27. package/esm2022/lib/components/loading_indicator/progress-spinner/progress-spinner.component.mjs +0 -65
  28. package/esm2022/lib/components/material-issues/material-issues.component.mjs +0 -23
  29. package/esm2022/lib/components/node-entry/node-entry.component.mjs +0 -35
  30. package/esm2022/lib/components/node-list/node-list.component.mjs +0 -112
  31. package/esm2022/lib/components/quality-matrix/quality_matrix.mjs +0 -413
  32. package/esm2022/lib/components/quality-matrix/scroll-marker.directive.mjs +0 -17
  33. package/esm2022/lib/config-helper.service.mjs +0 -32
  34. package/esm2022/lib/core/svg-icons.service.mjs +0 -44
  35. package/esm2022/lib/core/tooltip.service.mjs +0 -146
  36. package/esm2022/lib/counts-with-history/counts-with-history.component.mjs +0 -203
  37. package/esm2022/lib/java-api/api/api.mjs +0 -12
  38. package/esm2022/lib/java-api/api/authProxyController.service.mjs +0 -107
  39. package/esm2022/lib/java-api/api/collectionAPI.service.mjs +0 -409
  40. package/esm2022/lib/java-api/api/editorsAPI.service.mjs +0 -157
  41. package/esm2022/lib/java-api/api/filterAPI.service.mjs +0 -237
  42. package/esm2022/lib/java-api/api/replicationSourceAPI.service.mjs +0 -230
  43. package/esm2022/lib/java-api/api.base.service.mjs +0 -66
  44. package/esm2022/lib/java-api/api.module.mjs +0 -40
  45. package/esm2022/lib/java-api/configuration.mjs +0 -103
  46. package/esm2022/lib/java-api/encoder.mjs +0 -19
  47. package/esm2022/lib/java-api/index.mjs +0 -7
  48. package/esm2022/lib/java-api/model/collectionWithMissingAttributes.mjs +0 -11
  49. package/esm2022/lib/java-api/model/count.mjs +0 -11
  50. package/esm2022/lib/java-api/model/eduCollection.mjs +0 -11
  51. package/esm2022/lib/java-api/model/filter.mjs +0 -2
  52. package/esm2022/lib/java-api/model/filterValue.mjs +0 -11
  53. package/esm2022/lib/java-api/model/materialCountDto.mjs +0 -2
  54. package/esm2022/lib/java-api/model/materialCountFilter.mjs +0 -11
  55. package/esm2022/lib/java-api/model/materialWithMissingAttributes.mjs +0 -11
  56. package/esm2022/lib/java-api/model/matrixRowWithCounts.mjs +0 -2
  57. package/esm2022/lib/java-api/model/matrixWithCounts.mjs +0 -2
  58. package/esm2022/lib/java-api/model/missingAttributeResult.mjs +0 -2
  59. package/esm2022/lib/java-api/model/models.mjs +0 -16
  60. package/esm2022/lib/java-api/model/qualityMatrix.mjs +0 -2
  61. package/esm2022/lib/java-api/model/qualityMatrixHeader.mjs +0 -11
  62. package/esm2022/lib/java-api/model/qualityMatrixReplicationSourceCounts.mjs +0 -11
  63. package/esm2022/lib/java-api/model/qualityMatrixRow.mjs +0 -2
  64. package/esm2022/lib/java-api/param.mjs +0 -2
  65. package/esm2022/lib/java-api/variables.mjs +0 -9
  66. package/esm2022/lib/meta-api.service.mjs +0 -77
  67. package/esm2022/lib/ng-meta-widgets-lib.module.mjs +0 -198
  68. package/esm2022/lib/node-image-url.pipe.mjs +0 -29
  69. package/esm2022/lib/tree-collection-details/tree-collection-details.component.mjs +0 -87
  70. package/esm2022/lib/tree-license/tree-license.component.mjs +0 -136
  71. package/esm2022/lib/tree-search-counts/inline-worker.mjs +0 -102
  72. package/esm2022/lib/tree-search-counts/tree-search-counts.component.mjs +0 -209
  73. package/esm2022/lib/wrap-observable.pipe.mjs +0 -21
  74. package/esm2022/ngx-edu-sharing-metaqs2.mjs +0 -5
  75. package/esm2022/public-api.mjs +0 -18
  76. package/esm2022/web-components.mjs +0 -36
@@ -1,413 +0,0 @@
1
- import { Component, computed, DestroyRef, ElementRef, HostListener, inject, Input, signal, ViewChild, ViewChildren, } from '@angular/core';
2
- import { BehaviorSubject, combineLatest, of, Subject } from 'rxjs';
3
- import { filter, finalize, map, switchMap, take, tap, throttleTime } from 'rxjs/operators';
4
- import { MetaApiService } from '../../meta-api.service';
5
- import { FormControl, FormGroup, FormRecord, ReactiveFormsModule } from '@angular/forms';
6
- import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
7
- import { EditorialLinkService } from '../editorial-link-service/editorial-link.service';
8
- import { DateTime } from 'luxon';
9
- import { KeyValuePipe, NgClass, NgForOf, NgIf } from '@angular/common';
10
- import { Overlay } from '@angular/cdk/overlay';
11
- import { DonutChartTooltipComponent, transformDonutChartData, } from '../donut-chart-tooltip/donut-chart-tooltip.component';
12
- import { centerPositionStrategy, TooltipService } from '../../core/tooltip.service';
13
- import { MatCard, MatCardContent, MatCardHeader, MatCardTitle } from '@angular/material/card';
14
- import { QualityMatrixFilterComponent } from '../filter/quality-matrix-filter.component';
15
- import { TranslateModule } from '@ngx-translate/core';
16
- import { MatIcon } from '@angular/material/icon';
17
- import { DatepickerComponent } from '../filter/datepicker/datepicker.component';
18
- import { MatSlideToggle } from '@angular/material/slide-toggle';
19
- import { ProgressSpinnerComponent } from '../loading_indicator/progress-spinner/progress-spinner.component';
20
- import { MatCell, MatCellDef, MatColumnDef, MatHeaderCell, MatHeaderCellDef, MatHeaderRow, MatHeaderRowDef, MatRow, MatRowDef, MatTable, } from '@angular/material/table';
21
- import { MatTooltip } from '@angular/material/tooltip';
22
- import { MatButton, MatIconButton } from '@angular/material/button';
23
- import { DonutChartComponent } from '../donut-chart/donut-chart.component';
24
- import { ScrollMarkerDirective } from './scroll-marker.directive';
25
- import * as i0 from "@angular/core";
26
- import * as i1 from "@ngx-translate/core";
27
- import * as i2 from "@angular/forms";
28
- const openCloseTimeout = 150;
29
- export class QualityMatrixComponent {
30
- constructor() {
31
- this.ref = inject(ElementRef);
32
- this.destroyRef = inject(DestroyRef);
33
- this.metaApi = inject(MetaApiService);
34
- this.linkService = inject(EditorialLinkService);
35
- this.overlay = inject(Overlay);
36
- this.tooltipService = inject(TooltipService);
37
- this.scrollLeft = signal(0);
38
- this.scrollWidth = signal(0);
39
- this.isLeftScrollable = computed(() => this.scrollLeft() > 0);
40
- this.isRightScrollable = computed(() => {
41
- const viewport = globalThis.visualViewport;
42
- if (!viewport) {
43
- return;
44
- }
45
- const viewportWidth = viewport.width;
46
- return this.scrollWidth() > viewportWidth && this.scrollLeft() + viewportWidth < this.scrollWidth();
47
- });
48
- this.filteredColumns = computed(() => {
49
- const allColumns = this.allColumns();
50
- const isOnlySourcesWithMaterialEnabled = this.isOnlySourcesWithMaterialEnabled();
51
- if (isOnlySourcesWithMaterialEnabled) {
52
- return allColumns.filter((c) => c.total > 0);
53
- }
54
- else {
55
- return allColumns;
56
- }
57
- });
58
- this.datatableColumns = computed(() => {
59
- return this.filteredColumns()
60
- .filter((c) => c.level !== 0)
61
- .map((c) => c.id);
62
- });
63
- this.isHistoricalDataEnabledFormField = new FormControl({ value: false, disabled: true });
64
- this.isHistoricalDataEnabled = toSignal(this.isHistoricalDataEnabledFormField.valueChanges);
65
- this.isOnlySourcesWithMaterialEnabledFormField = new FormControl({ value: true, disabled: true });
66
- this.isOnlySourcesWithMaterialEnabled = toSignal(this.isOnlySourcesWithMaterialEnabledFormField.valueChanges);
67
- this.categoryControls = new FormRecord({});
68
- this.loadingCount = signal(0);
69
- this.isLoading = computed(() => this.loadingCount() > 0);
70
- this.categoryFilterValues = new Map();
71
- this.refresh$ = new Subject();
72
- this.recentQualityMatrix$ = new BehaviorSubject({ columns: [], rows: [] });
73
- this.recentQualityMatrix = toSignal(this.recentQualityMatrix$, {
74
- initialValue: { columns: [], rows: [] },
75
- });
76
- this.pastQualityMatrix$ = new BehaviorSubject({ columns: [], rows: [] });
77
- this.range = new FormGroup({
78
- start: new FormControl(),
79
- end: new FormControl(),
80
- });
81
- this.tooltips = new Map();
82
- this.DateTime = DateTime;
83
- this.pageTitle = 'Quality Matrix';
84
- this.allColumns = computed(() => this.recentQualityMatrix()?.columns.toSorted((a, b) => a.label.localeCompare(b.label)));
85
- this.sourceColumns = computed(() => this.datatableColumns().map((c) => c + '_source'));
86
- this.currentColumns = computed(() => this.datatableColumns().map((c) => c + '_current'));
87
- this.pastColumns = computed(() => this.datatableColumns().map((c) => c + '_past'));
88
- this.allDataColumns = computed(() => {
89
- const pastColumns = this.pastColumns();
90
- const currentColumns = this.currentColumns();
91
- if (this.isHistoricalDataEnabled()) {
92
- return currentColumns.flatMap((e, i) => [pastColumns[i], e]);
93
- }
94
- return currentColumns;
95
- });
96
- }
97
- ngAfterViewChecked() {
98
- this.onScroll();
99
- }
100
- filterIdent(_index, item) {
101
- return item.key;
102
- }
103
- loadData(categories, date) {
104
- return of(undefined).pipe(tap(() => this.loadingCount.update((it) => it + 1)), switchMap(() => this.metaApi
105
- .getQualityMatrixWithFiltersV2([
106
- ...this.transformCategoryFilters(categories),
107
- {
108
- field: 'asOf',
109
- values: [
110
- {
111
- id: date.toISO({
112
- includeOffset: false,
113
- }),
114
- label: '',
115
- },
116
- ],
117
- },
118
- ])
119
- .pipe(take(1))), finalize(() => this.loadingCount.update((it) => it - 1)));
120
- }
121
- transformCategoryFilters(categories) {
122
- return Object.entries(categories)
123
- .filter((pair) => !!pair[1]?.length)
124
- .map(([field, values]) => ({
125
- field,
126
- values: values.map((value) => ({ id: value, label: '' })),
127
- }));
128
- }
129
- loadPastData(categories, date) {
130
- return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));
131
- }
132
- loadCurrentData(categories, date) {
133
- return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));
134
- }
135
- openEditLink(source, issue) {
136
- return this.linkService.openByReplicationsourceAndIssueTypeWithFilters(source, issue, this.categoryControls.value);
137
- }
138
- // see https://github.com/angular/components/issues/8361#issuecomment-345804954
139
- columnIdent(_index, col) {
140
- return col.id;
141
- }
142
- refresh() {
143
- this.refresh$.next();
144
- }
145
- getPastColumn(columnId) {
146
- return this.pastQualityMatrix$.value?.columns.find((c) => c.id === columnId);
147
- }
148
- ngOnInit() {
149
- this.loadingCount.update((it) => it + 1);
150
- this.metaApi
151
- .getCategoryFilters()
152
- .pipe(take(1), tap((filters) => {
153
- filters.forEach((filter) => {
154
- // do not emit the event when setting the control to prevent unnecessary valueChanges events
155
- this.categoryControls.addControl(filter.field, new FormControl([], { nonNullable: true }), {
156
- emitEvent: false,
157
- });
158
- this.categoryFilterValues.set(filter.field, filter.values.toSorted((a, b) => {
159
- return a.label.localeCompare(b.label);
160
- }));
161
- });
162
- // emit the current value of the form group once to trigger the valueChanges observable
163
- this.categoryControls.setValue(this.categoryControls.value, {
164
- emitEvent: true,
165
- });
166
- }), takeUntilDestroyed(this.destroyRef), finalize(() => this.loadingCount.update((it) => it - 1)))
167
- .subscribe();
168
- this.loadingCount.update((it) => it + 1);
169
- this.metaApi
170
- .getTimerangeFilter()
171
- .pipe(filter((rangeFilter) => !!rangeFilter), tap((rangeFilter) => {
172
- const startDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label, {
173
- zone: 'utc',
174
- });
175
- const endDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeEnd')?.label, {
176
- zone: 'utc',
177
- });
178
- this.range.setControl('start', new FormControl(startDate, { nonNullable: true }), {
179
- emitEvent: false,
180
- });
181
- this.range.setControl('end', new FormControl(endDate, { nonNullable: true }), {
182
- emitEvent: true,
183
- });
184
- }), takeUntilDestroyed(this.destroyRef), finalize(() => this.loadingCount.update((it) => it - 1)))
185
- .subscribe();
186
- combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$])
187
- .pipe(filter((data) => !!data[0] && !!data[1].end?.isValid), throttleTime(500), map(([categories, range]) => [categories, range.end]), switchMap(([categories, end]) => this.loadCurrentData(categories, end)), tap((currentQm) => {
188
- this.recentQualityMatrix$.next(currentQm);
189
- }), takeUntilDestroyed(this.destroyRef))
190
- .subscribe();
191
- let pastDataSubscription;
192
- const pastData$ = combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$]).pipe(filter((data) => !!data[0] && !!data[1].start?.isValid), throttleTime(500), map(([categories, range]) => [categories, range.start]), switchMap(([categories, start]) => this.loadPastData(categories, start)), tap((pastQm) => {
193
- this.pastQualityMatrix$.next(pastQm);
194
- }), takeUntilDestroyed(this.destroyRef));
195
- this.isHistoricalDataEnabledFormField.valueChanges
196
- .pipe(tap((isEnabled) => {
197
- if (isEnabled) {
198
- pastDataSubscription = pastData$.subscribe();
199
- this.categoryControls.setValue(this.categoryControls.value, {
200
- emitEvent: true,
201
- });
202
- this.range.setValue(this.range.value, {
203
- emitEvent: true,
204
- });
205
- this.refresh$.next();
206
- }
207
- else {
208
- pastDataSubscription?.unsubscribe();
209
- }
210
- }), takeUntilDestroyed(this.destroyRef))
211
- .subscribe();
212
- this.refresh$.next();
213
- this.isOnlySourcesWithMaterialEnabledFormField.setValue(this.isOnlySourcesWithMaterialEnabledFormField.value, {
214
- emitEvent: true,
215
- });
216
- }
217
- getTrend(row, column) {
218
- if (this.pastQualityMatrix$.value === null || this.pastQualityMatrix$.value.columns.length === 0) {
219
- return {};
220
- }
221
- const past_row = this.pastQualityMatrix$?.value.rows.find((pr) => pr.meta.id == row.meta.id);
222
- const past_column = this.pastQualityMatrix$.value.columns.find((c) => c.id == column.id);
223
- if (!past_row || !past_column) {
224
- return {};
225
- }
226
- const delta = Math.floor((100 * row.counts[column.id].sufficient) / column.total -
227
- (100 * past_row.counts[column.id].sufficient) / past_column.total);
228
- const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';
229
- return { delta, trend };
230
- }
231
- getDonutSlices(row, column) {
232
- const sufficientTotal = row.counts[column.id].sufficient;
233
- const insufficientTotal = row.counts[column.id].insufficient;
234
- const missingTotal = column.total - sufficientTotal - insufficientTotal;
235
- let sufficientFillRate = 0;
236
- let insufficientFillRate = 0;
237
- //Der Wert berechnet sich aus dem total_count - without_description_count.
238
- if (column.total !== 0) {
239
- sufficientFillRate = Math.round((100 * sufficientTotal) / column.total);
240
- insufficientFillRate = Math.round((100 * insufficientTotal) / column.total);
241
- }
242
- const rest = 100 - (sufficientFillRate + insufficientFillRate); // a.k.a. "missing"
243
- return [
244
- {
245
- id: 3,
246
- percent: rest,
247
- color: '#D8EDFC',
248
- label: `${rest} % (${missingTotal})`,
249
- verboseLabel: `${missingTotal} Metadatens\u00e4tze fehlen`,
250
- },
251
- {
252
- id: 1,
253
- percent: insufficientFillRate,
254
- color: '#C20808',
255
- label: `${insufficientFillRate} % (${insufficientTotal})`,
256
- verboseLabel: `${insufficientTotal} Metadatens\u00e4tze sind unzureichend ausgef\u00fcllt`,
257
- },
258
- {
259
- id: 2,
260
- percent: sufficientFillRate,
261
- color: '#4ABEFF',
262
- label: `${sufficientFillRate} % (${sufficientTotal})`,
263
- verboseLabel: `${sufficientTotal} Metadatens\u00e4tze sind gut ausgef\u00fcllt`,
264
- },
265
- ];
266
- }
267
- showTooltip(row, column, cell) {
268
- if (row.counts[column.id].sufficient + row.counts[column.id].insufficient === 0) {
269
- return;
270
- }
271
- if (this.tooltips.has(cell)) {
272
- this.tooltips.get(cell).cancelCloseAction();
273
- return;
274
- }
275
- const tooltipRef = this.tooltipService.create(DonutChartTooltipComponent, {
276
- closeTimeout: openCloseTimeout,
277
- openTimeout: openCloseTimeout,
278
- data: {
279
- label: column.label,
280
- data: transformDonutChartData(this.getDonutSlices(row, column)),
281
- },
282
- overlayConfig: {
283
- hasBackdrop: false,
284
- positionStrategy: this.overlay.position().flexibleConnectedTo(cell).withPositions(centerPositionStrategy),
285
- disposeOnNavigation: true,
286
- panelClass: 'donut-chart-tooltip',
287
- },
288
- onClose: () => {
289
- this.tooltips.delete(cell);
290
- },
291
- });
292
- tooltipRef.open();
293
- this.tooltips.set(cell, tooltipRef);
294
- }
295
- hideTooltip(cell) {
296
- this.tooltips.get(cell)?.markForClose();
297
- }
298
- findHorizontalScrollContainer(element) {
299
- if (element.scrollWidth > element.clientWidth && ['auto', 'scroll'].includes(getComputedStyle(element).overflowX)) {
300
- // find the nearest horizontally scrollable element
301
- return element;
302
- }
303
- else {
304
- if (!element.parentElement) {
305
- return document.scrollingElement;
306
- }
307
- return this.findHorizontalScrollContainer(element.parentElement);
308
- }
309
- }
310
- scrollToRight() {
311
- const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
312
- if (!scrollingElement) {
313
- return;
314
- }
315
- const viewport = globalThis.visualViewport;
316
- if (!viewport) {
317
- return;
318
- }
319
- const viewportWidth = viewport.width;
320
- const lastVisible = this.scrollMarkers.findLast((marker) => marker.element.nativeElement.getBoundingClientRect().left < viewportWidth);
321
- if (!lastVisible) {
322
- return;
323
- }
324
- const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;
325
- scrollingElement.scrollBy({
326
- left: lastVisible.element.nativeElement.getBoundingClientRect().left - leftColumnWidth,
327
- behavior: 'smooth',
328
- });
329
- }
330
- scrollToLeft() {
331
- const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
332
- if (!scrollingElement) {
333
- return;
334
- }
335
- const viewport = globalThis.visualViewport;
336
- if (!viewport) {
337
- return;
338
- }
339
- const viewportWidth = viewport.width;
340
- const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;
341
- const firstVisible = this.scrollMarkers.find((marker) => marker.element.nativeElement.getBoundingClientRect().right > leftColumnWidth);
342
- if (!firstVisible) {
343
- return;
344
- }
345
- scrollingElement.scrollBy({
346
- left: firstVisible.element.nativeElement.getBoundingClientRect().right - viewportWidth,
347
- behavior: 'smooth',
348
- });
349
- }
350
- onScroll() {
351
- const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
352
- if (!scrollingElement) {
353
- return;
354
- }
355
- this.scrollLeft.set(scrollingElement.scrollLeft);
356
- this.scrollWidth.set(scrollingElement.scrollWidth);
357
- }
358
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: QualityMatrixComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
359
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: QualityMatrixComponent, isStandalone: true, selector: "metaqs2-quality-matrix-v2", inputs: { pageTitle: "pageTitle" }, host: { listeners: { "window:scroll": "onScroll()", "window:resize": "onScroll()" } }, providers: [TooltipService], viewQueries: [{ propertyName: "issueTypeHeader", first: true, predicate: ["issueType"], descendants: true, read: ElementRef, static: true }, { propertyName: "scrollMarkers", predicate: ScrollMarkerDirective, descendants: true }], ngImport: i0, template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n </mat-card-header>\n <mat-card-content class=\"toolbar\">\n <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n [options]=\"categoryFilterValues.get(filter.key)\"\n [inputFormControl]=\"filter.value\"\n [label] = \"'filter.' + filter.key | translate\" />\n <div class=\"actionbar\">\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n show only sources with material\n </mat-slide-toggle>\n <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n show historical data\n </mat-slide-toggle>\n <button\n mat-icon-button\n color=\"primary\"\n [disabled]=\"isLoading()\"\n (click)=\"refresh()\"\n matTooltip=\"Aktualisiere den IST-Stand\"\n matTooltipShowDelay=\"500\"\n >\n <mat-icon>refresh</mat-icon>\n </button>\n <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n </button>\n <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n </button>\n </div>\n </div>\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]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"row-header\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n >\n {{\"quality_matrix.\" + row.meta.label | translate }}\n </td>\n </ng-container>\n <!-- one column for the source -->\n <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- current Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n <div>\n <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n </div>\n </a>\n <ng-template #emptyDiv>\n <div>-</div>\n </ng-template>\n </td>\n </ng-container>\n <!-- /current Data Columns -->\n <!-- past Data Columns -->\n <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n <th class=\"historic-data-cell\" mat-header-cell\n *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n <br>{{ getPastColumn(column.id)?.total }}\n </th>\n <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n <ng-container *ngIf=\"getTrend(row, column) as trend\">\n <span [ngClass]=\"trend.trend\">\n <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n {{ trend.delta }}\n </span>\n </ng-container>\n\n </td>\n </ng-container>\n </ng-container>\n <!-- /past Data Columns -->\n\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}mat-card-actions>mat-slide-toggle{margin-left:20px}.row{flex-grow:1;display:flex;align-items:center}table{width:100%;overflow-y:auto}table .scroll{flex-grow:1;overflow-y:auto}table th{vertical-align:top;padding:.375rem}table th:not(:last-child){border-right:1px solid #e4e4e4}table tr:first-child>th:first-child{text-align:left}table td:first-child{text-align:left}.cell{flex:1}.mat-mdc-cell{padding-top:5px}.mat-mdc-cell,.mat-mdc-header-cell{padding-left:5px;text-align:center}.mat-mdc-cell a.chart-container,.mat-mdc-header-cell a.chart-container{display:flex;justify-content:center}.mat-mdc-cell .trending_up,.mat-mdc-header-cell .trending_up{color:#4abeff}.mat-mdc-cell .trending_down,.mat-mdc-header-cell .trending_down{color:#c20808}.recent-data-cell,.mat-column-row-header{border-right:1px solid #e4e4e4}.mat-header-cell-label{color:#000;padding-left:30px;padding-right:30px;justify-content:left;border-bottom:1px solid #e4e4e4;font-style:normal;font-weight:500;font-size:110%;background:#ddf0fb;text-align:left}.mat-header-cell-label>div{display:flex;font-weight:700;font-style:normal;justify-content:left}.mat-header-cell-small{padding-left:5px;padding-right:5px;justify-content:left;font-style:normal;font-weight:400;font-size:105%;background:#ddf0fb;border-top:1px solid #e4e4e4;border-bottom:1px solid #e4e4e4}.mat-mdc-cell a{cursor:pointer;display:flex;align-items:center;justify-content:left}.mat-mdc-cell a>mat-icon{padding-right:5px;font-size:22px;height:20px;opacity:.75}.mat-mdc-cell a>mat-icon:hover,.mat-mdc-cell a>mat-icon:focus{text-decoration:none}.mat-mdc-cell .number{display:flex;height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header-sum{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;border-right:5px solid #bdbdbd}.mat-mdc-card{width:100%;box-shadow:none}.mat-mdc-card ::ng-deep mat-card-header{align-items:center}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text{flex-grow:1;margin:0}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title{padding-top:20px;font-size:120%}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title .count{font-size:80%;color:#888}.mat-mdc-card .source-header{font-style:normal;font-weight:400;font-size:20px;height:30px;width:60%}.mat-mdc-card .source-header .robot-explanation{font-size:15px}.mat-mdc-card .scroll{flex-grow:1;overflow-y:auto}.mat-header-cell-scope{color:#545454}.mat-mdc-header-row{height:70px;max-height:70px;min-height:56px}:host{overflow:auto}:host ::ng-deep thead{border:10px solid #ff0000;position:relative;z-index:1}.mat-cell-level-1{padding-left:20px!important}.mat-cell-level-2{padding-left:40px!important}.mat-cell-level-3{padding-left:60px!important}.mat-cell-level-4{padding-left:80px!important}.mat-cell-level-5{padding-left:100px!important}.mat-cell-level-6{padding-left:120px!important}.mat-cell-level-7{padding-left:140px!important}.mat-cell-level-8{padding-left:160px!important}.actionbar{flex:1 1 auto;display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}.scroll-controller{position:fixed;right:.5rem;z-index:100;display:flex;flex-direction:row;justify-content:space-between;align-items:center;gap:.5rem}.scroll-button{min-width:0;padding-left:.25rem;padding-right:.25rem;text-align:center;height:1.75rem}.toolbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:stretch;gap:.5rem}\n"], dependencies: [{ kind: "component", type: MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "component", type: MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "directive", type: MatCardContent, selector: "mat-card-content" }, { kind: "component", type: QualityMatrixFilterComponent, selector: "metaqs2-qm-filter", inputs: ["options", "inputFormControl", "label", "multiple"], outputs: ["changedFilters"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }, { kind: "pipe", type: KeyValuePipe, name: "keyvalue" }, { kind: "component", type: MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: DatepickerComponent, selector: "metaqs2-datepicker", inputs: ["disabled", "inputGroup"] }, { 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: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: ProgressSpinnerComponent, selector: "metaqs2-progress-spinner", inputs: ["color", "diameter", "strokeWidth", "backdropEnabled", "positionGloballyCenter", "displayProgressSpinner"] }, { kind: "component", type: MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "directive", type: MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "directive", type: MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: MatCellDef, selector: "[matCellDef]" }, { kind: "component", type: MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "component", type: DonutChartComponent, selector: "metaqs2-donut-chart", inputs: ["radius", "viewBox", "borderSize", "strokeWidth", "data"] }, { kind: "directive", type: ScrollMarkerDirective, selector: "[metaqs2ScrollMarker]" }] }); }
360
- }
361
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: QualityMatrixComponent, decorators: [{
362
- type: Component,
363
- args: [{ selector: 'metaqs2-quality-matrix-v2', providers: [TooltipService], standalone: true, imports: [
364
- MatCard,
365
- MatCardHeader,
366
- MatCardTitle,
367
- MatCardContent,
368
- QualityMatrixFilterComponent,
369
- TranslateModule,
370
- KeyValuePipe,
371
- MatIcon,
372
- DatepickerComponent,
373
- MatSlideToggle,
374
- ReactiveFormsModule,
375
- ProgressSpinnerComponent,
376
- MatTable,
377
- MatTooltip,
378
- MatCell,
379
- MatColumnDef,
380
- MatHeaderCell,
381
- NgClass,
382
- MatHeaderRow,
383
- MatRow,
384
- MatHeaderRowDef,
385
- MatRowDef,
386
- MatHeaderCellDef,
387
- NgForOf,
388
- NgIf,
389
- MatCellDef,
390
- MatButton,
391
- MatIconButton,
392
- DonutChartComponent,
393
- ScrollMarkerDirective,
394
- ], template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n </mat-card-header>\n <mat-card-content class=\"toolbar\">\n <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n [options]=\"categoryFilterValues.get(filter.key)\"\n [inputFormControl]=\"filter.value\"\n [label] = \"'filter.' + filter.key | translate\" />\n <div class=\"actionbar\">\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n show only sources with material\n </mat-slide-toggle>\n <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n show historical data\n </mat-slide-toggle>\n <button\n mat-icon-button\n color=\"primary\"\n [disabled]=\"isLoading()\"\n (click)=\"refresh()\"\n matTooltip=\"Aktualisiere den IST-Stand\"\n matTooltipShowDelay=\"500\"\n >\n <mat-icon>refresh</mat-icon>\n </button>\n <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n </button>\n <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n </button>\n </div>\n </div>\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]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"row-header\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n >\n {{\"quality_matrix.\" + row.meta.label | translate }}\n </td>\n </ng-container>\n <!-- one column for the source -->\n <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- current Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n <div>\n <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n </div>\n </a>\n <ng-template #emptyDiv>\n <div>-</div>\n </ng-template>\n </td>\n </ng-container>\n <!-- /current Data Columns -->\n <!-- past Data Columns -->\n <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n <th class=\"historic-data-cell\" mat-header-cell\n *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n <br>{{ getPastColumn(column.id)?.total }}\n </th>\n <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n <ng-container *ngIf=\"getTrend(row, column) as trend\">\n <span [ngClass]=\"trend.trend\">\n <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n {{ trend.delta }}\n </span>\n </ng-container>\n\n </td>\n </ng-container>\n </ng-container>\n <!-- /past Data Columns -->\n\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}mat-card-actions>mat-slide-toggle{margin-left:20px}.row{flex-grow:1;display:flex;align-items:center}table{width:100%;overflow-y:auto}table .scroll{flex-grow:1;overflow-y:auto}table th{vertical-align:top;padding:.375rem}table th:not(:last-child){border-right:1px solid #e4e4e4}table tr:first-child>th:first-child{text-align:left}table td:first-child{text-align:left}.cell{flex:1}.mat-mdc-cell{padding-top:5px}.mat-mdc-cell,.mat-mdc-header-cell{padding-left:5px;text-align:center}.mat-mdc-cell a.chart-container,.mat-mdc-header-cell a.chart-container{display:flex;justify-content:center}.mat-mdc-cell .trending_up,.mat-mdc-header-cell .trending_up{color:#4abeff}.mat-mdc-cell .trending_down,.mat-mdc-header-cell .trending_down{color:#c20808}.recent-data-cell,.mat-column-row-header{border-right:1px solid #e4e4e4}.mat-header-cell-label{color:#000;padding-left:30px;padding-right:30px;justify-content:left;border-bottom:1px solid #e4e4e4;font-style:normal;font-weight:500;font-size:110%;background:#ddf0fb;text-align:left}.mat-header-cell-label>div{display:flex;font-weight:700;font-style:normal;justify-content:left}.mat-header-cell-small{padding-left:5px;padding-right:5px;justify-content:left;font-style:normal;font-weight:400;font-size:105%;background:#ddf0fb;border-top:1px solid #e4e4e4;border-bottom:1px solid #e4e4e4}.mat-mdc-cell a{cursor:pointer;display:flex;align-items:center;justify-content:left}.mat-mdc-cell a>mat-icon{padding-right:5px;font-size:22px;height:20px;opacity:.75}.mat-mdc-cell a>mat-icon:hover,.mat-mdc-cell a>mat-icon:focus{text-decoration:none}.mat-mdc-cell .number{display:flex;height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header-sum{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;border-right:5px solid #bdbdbd}.mat-mdc-card{width:100%;box-shadow:none}.mat-mdc-card ::ng-deep mat-card-header{align-items:center}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text{flex-grow:1;margin:0}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title{padding-top:20px;font-size:120%}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title .count{font-size:80%;color:#888}.mat-mdc-card .source-header{font-style:normal;font-weight:400;font-size:20px;height:30px;width:60%}.mat-mdc-card .source-header .robot-explanation{font-size:15px}.mat-mdc-card .scroll{flex-grow:1;overflow-y:auto}.mat-header-cell-scope{color:#545454}.mat-mdc-header-row{height:70px;max-height:70px;min-height:56px}:host{overflow:auto}:host ::ng-deep thead{border:10px solid #ff0000;position:relative;z-index:1}.mat-cell-level-1{padding-left:20px!important}.mat-cell-level-2{padding-left:40px!important}.mat-cell-level-3{padding-left:60px!important}.mat-cell-level-4{padding-left:80px!important}.mat-cell-level-5{padding-left:100px!important}.mat-cell-level-6{padding-left:120px!important}.mat-cell-level-7{padding-left:140px!important}.mat-cell-level-8{padding-left:160px!important}.actionbar{flex:1 1 auto;display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}.scroll-controller{position:fixed;right:.5rem;z-index:100;display:flex;flex-direction:row;justify-content:space-between;align-items:center;gap:.5rem}.scroll-button{min-width:0;padding-left:.25rem;padding-right:.25rem;text-align:center;height:1.75rem}.toolbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:stretch;gap:.5rem}\n"] }]
395
- }], propDecorators: { issueTypeHeader: [{
396
- type: ViewChild,
397
- args: ['issueType', {
398
- read: ElementRef,
399
- static: true,
400
- }]
401
- }], scrollMarkers: [{
402
- type: ViewChildren,
403
- args: [ScrollMarkerDirective]
404
- }], pageTitle: [{
405
- type: Input
406
- }], onScroll: [{
407
- type: HostListener,
408
- args: ['window:scroll']
409
- }, {
410
- type: HostListener,
411
- args: ['window:resize']
412
- }] } });
413
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quality_matrix.js","sourceRoot":"","sources":["../../../../../../projects/ng-meta-widgets-lib/src/lib/components/quality-matrix/quality_matrix.ts","../../../../../../projects/ng-meta-widgets-lib/src/lib/components/quality-matrix/quality-matrix.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  computed,\n  DestroyRef,\n  ElementRef,\n  HostListener,\n  inject,\n  Input,\n  OnInit,\n  signal,\n  ViewChild,\n  ViewChildren,\n} from '@angular/core';\nimport { BehaviorSubject, combineLatest, Observable, of, Subject, Unsubscribable } from 'rxjs';\nimport { filter, finalize, map, switchMap, take, tap, throttleTime } from 'rxjs/operators';\nimport { MetaApiService } from '../../meta-api.service';\nimport { Filter, FilterValue, QualityMatrix, QualityMatrixHeader, QualityMatrixRow } from '../../java-api';\nimport { FormControl, FormGroup, FormRecord, ReactiveFormsModule } from '@angular/forms';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { EditorialLinkService } from '../editorial-link-service/editorial-link.service';\nimport { DateTime } from 'luxon';\nimport { DonutSlice } from '../donut-chart/donut-chart.model';\nimport { KeyValue, KeyValuePipe, NgClass, NgForOf, NgIf } from '@angular/common';\nimport { Overlay } from '@angular/cdk/overlay';\nimport {\n  DonutChartTooltipComponent,\n  transformDonutChartData,\n} from '../donut-chart-tooltip/donut-chart-tooltip.component';\nimport { centerPositionStrategy, TooltipRef, TooltipService } from '../../core/tooltip.service';\nimport { MatCard, MatCardContent, MatCardHeader, MatCardTitle } from '@angular/material/card';\nimport { QualityMatrixFilterComponent } from '../filter/quality-matrix-filter.component';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { MatIcon } from '@angular/material/icon';\nimport { DatepickerComponent } from '../filter/datepicker/datepicker.component';\nimport { MatSlideToggle } from '@angular/material/slide-toggle';\nimport { ProgressSpinnerComponent } from '../loading_indicator/progress-spinner/progress-spinner.component';\nimport {\n  MatCell,\n  MatCellDef,\n  MatColumnDef,\n  MatHeaderCell,\n  MatHeaderCellDef,\n  MatHeaderRow,\n  MatHeaderRowDef,\n  MatRow,\n  MatRowDef,\n  MatTable,\n} from '@angular/material/table';\nimport { MatTooltip } from '@angular/material/tooltip';\nimport { MatButton, MatIconButton } from '@angular/material/button';\nimport { DonutChartComponent } from '../donut-chart/donut-chart.component';\nimport { ScrollMarkerDirective } from './scroll-marker.directive';\n\nconst openCloseTimeout = 150;\n\n@Component({\n  selector: 'metaqs2-quality-matrix-v2',\n  providers: [TooltipService],\n  standalone: true,\n  templateUrl: './quality-matrix.html',\n  imports: [\n    MatCard,\n    MatCardHeader,\n    MatCardTitle,\n    MatCardContent,\n    QualityMatrixFilterComponent,\n    TranslateModule,\n    KeyValuePipe,\n    MatIcon,\n    DatepickerComponent,\n    MatSlideToggle,\n    ReactiveFormsModule,\n    ProgressSpinnerComponent,\n    MatTable,\n    MatTooltip,\n    MatCell,\n    MatColumnDef,\n    MatHeaderCell,\n    NgClass,\n    MatHeaderRow,\n    MatRow,\n    MatHeaderRowDef,\n    MatRowDef,\n    MatHeaderCellDef,\n    NgForOf,\n    NgIf,\n    MatCellDef,\n    MatButton,\n    MatIconButton,\n    DonutChartComponent,\n    ScrollMarkerDirective,\n  ],\n  styleUrls: ['./quality_matrix.scss'],\n})\nexport class QualityMatrixComponent implements OnInit, AfterViewChecked {\n  protected readonly ref = inject(ElementRef);\n  protected readonly destroyRef = inject(DestroyRef);\n  protected readonly metaApi = inject(MetaApiService);\n  protected readonly linkService = inject(EditorialLinkService);\n  protected readonly overlay = inject(Overlay);\n  protected readonly tooltipService = inject(TooltipService);\n  private readonly scrollLeft = signal(0);\n  private readonly scrollWidth = signal(0);\n  public readonly isLeftScrollable = computed(() => this.scrollLeft() > 0);\n  public readonly isRightScrollable = computed(() => {\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    return this.scrollWidth() > viewportWidth && this.scrollLeft() + viewportWidth < this.scrollWidth();\n  });\n  @ViewChild('issueType', {\n    read: ElementRef,\n    static: true,\n  })\n  private readonly issueTypeHeader!: ElementRef<HTMLTableCellElement>;\n  readonly filteredColumns = computed<Array<QualityMatrixHeader>>(() => {\n    const allColumns = this.allColumns();\n    const isOnlySourcesWithMaterialEnabled = this.isOnlySourcesWithMaterialEnabled();\n    if (isOnlySourcesWithMaterialEnabled) {\n      return allColumns.filter((c) => c.total > 0);\n    } else {\n      return allColumns;\n    }\n  });\n  readonly datatableColumns = computed(() => {\n    return this.filteredColumns()\n      .filter((c) => c.level !== 0)\n      .map((c) => c.id);\n  });\n  readonly isHistoricalDataEnabledFormField = new FormControl({ value: false, disabled: true });\n  readonly isHistoricalDataEnabled = toSignal(this.isHistoricalDataEnabledFormField.valueChanges);\n  readonly isOnlySourcesWithMaterialEnabledFormField = new FormControl({ value: true, disabled: true });\n  readonly isOnlySourcesWithMaterialEnabled = toSignal(this.isOnlySourcesWithMaterialEnabledFormField.valueChanges);\n  readonly categoryControls = new FormRecord<FormControl<Array<string>>>({});\n  readonly loadingCount = signal(0);\n  readonly isLoading = computed(() => this.loadingCount() > 0);\n  @ViewChildren(ScrollMarkerDirective)\n  private readonly scrollMarkers!: ReadonlyArray<ScrollMarkerDirective>;\n  readonly categoryFilterValues: Map<string, FilterValue[]> = new Map();\n  protected readonly refresh$ = new Subject<void>();\n  readonly recentQualityMatrix$ = new BehaviorSubject<QualityMatrix>({ columns: [], rows: [] } as QualityMatrix);\n  readonly recentQualityMatrix = toSignal(this.recentQualityMatrix$, {\n    initialValue: { columns: [], rows: [] } as QualityMatrix,\n  });\n  readonly pastQualityMatrix$ = new BehaviorSubject<QualityMatrix | null>({ columns: [], rows: [] } as QualityMatrix);\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(),\n  });\n  private readonly tooltips = new Map<HTMLElement, TooltipRef<DonutChartTooltipComponent>>();\n  protected readonly DateTime = DateTime;\n  @Input()\n  pageTitle: string = 'Quality Matrix';\n\n  readonly allColumns = computed(() =>\n    this.recentQualityMatrix()?.columns.toSorted((a, b) => a.label.localeCompare(b.label))\n  );\n\n  ngAfterViewChecked(): void {\n    this.onScroll();\n  }\n\n  filterIdent(_index: number, item: KeyValue<string, FormControl<string[]>>) {\n    return item.key;\n  }\n\n  sourceColumns = computed(() => this.datatableColumns().map((c) => c + '_source'));\n\n  currentColumns = computed(() => this.datatableColumns().map((c) => c + '_current'));\n\n  pastColumns = computed(() => this.datatableColumns().map((c) => c + '_past'));\n\n  allDataColumns = computed(() => {\n    const pastColumns = this.pastColumns();\n    const currentColumns = this.currentColumns();\n    if (this.isHistoricalDataEnabled()) {\n      return currentColumns.flatMap((e, i) => [pastColumns[i], e]);\n    }\n    return currentColumns;\n  });\n\n  private loadData(categories: { [key: string]: Array<string> }, date: DateTime<true>): Observable<QualityMatrix> {\n    return of(undefined).pipe(\n      tap(() => this.loadingCount.update((it) => it + 1)),\n      switchMap(() =>\n        this.metaApi\n          .getQualityMatrixWithFiltersV2([\n            ...this.transformCategoryFilters(categories),\n            {\n              field: 'asOf',\n              values: [\n                {\n                  id: date.toISO({\n                    includeOffset: false,\n                  }) as string,\n                  label: '',\n                },\n              ],\n            },\n          ])\n          .pipe(take(1))\n      ),\n      finalize(() => this.loadingCount.update((it) => it - 1))\n    );\n  }\n\n  private transformCategoryFilters(categories: { [key: string]: Array<string> }): Filter[] {\n    return Object.entries(categories)\n      .filter((pair): pair is [string, Array<string>] => !!pair[1]?.length)\n      .map(([field, values]) => ({\n        field,\n        values: values.map((value) => ({ id: value, label: '' })),\n      }));\n  }\n\n  private loadPastData(categories: any, date: DateTime<true>) {\n    return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));\n  }\n\n  private loadCurrentData(categories: any, date: DateTime<true>) {\n    return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));\n  }\n\n  openEditLink(source: string, issue: string) {\n    return this.linkService.openByReplicationsourceAndIssueTypeWithFilters(source, issue, this.categoryControls.value);\n  }\n\n  // see https://github.com/angular/components/issues/8361#issuecomment-345804954\n  columnIdent(_index: number, col: QualityMatrixHeader) {\n    return col.id;\n  }\n\n  refresh() {\n    this.refresh$.next();\n  }\n\n  getPastColumn(columnId: string) {\n    return this.pastQualityMatrix$.value?.columns.find((c) => c.id === columnId);\n  }\n\n  public ngOnInit() {\n    this.loadingCount.update((it) => it + 1);\n    this.metaApi\n      .getCategoryFilters()\n      .pipe(\n        take(1),\n        tap((filters) => {\n          filters.forEach((filter) => {\n            // do not emit the event when setting the control to prevent unnecessary valueChanges events\n            this.categoryControls.addControl(filter.field, new FormControl([], { nonNullable: true }), {\n              emitEvent: false,\n            });\n            this.categoryFilterValues.set(\n              filter.field,\n              filter.values.toSorted((a, b) => {\n                return a.label.localeCompare(b.label);\n              })\n            );\n          });\n          // emit the current value of the form group once to trigger the valueChanges observable\n          this.categoryControls.setValue(this.categoryControls.value as { [key: string]: string[] }, {\n            emitEvent: true,\n          });\n        }),\n        takeUntilDestroyed(this.destroyRef),\n        finalize(() => this.loadingCount.update((it) => it - 1))\n      )\n      .subscribe();\n\n    this.loadingCount.update((it) => it + 1);\n    this.metaApi\n      .getTimerangeFilter()\n      .pipe(\n        filter((rangeFilter) => !!rangeFilter),\n        tap((rangeFilter) => {\n          const startDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label!, {\n            zone: 'utc',\n          });\n          const endDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeEnd')?.label!, {\n            zone: 'utc',\n          });\n          this.range.setControl('start', new FormControl<DateTime>(startDate, { nonNullable: true }), {\n            emitEvent: false,\n          });\n          this.range.setControl('end', new FormControl<DateTime>(endDate, { nonNullable: true }), {\n            emitEvent: true,\n          });\n        }),\n        takeUntilDestroyed(this.destroyRef),\n        finalize(() => this.loadingCount.update((it) => it - 1))\n      )\n      .subscribe();\n\n    combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$])\n      .pipe(\n        filter(\n          (data): data is [{ [key: string]: Array<string> }, { end: DateTime<true> }, void] =>\n            !!data[0] && !!data[1].end?.isValid\n        ),\n        throttleTime(500),\n        map(([categories, range]) => [categories, range.end] as const),\n        switchMap(([categories, end]) => this.loadCurrentData(categories, end)),\n        tap((currentQm) => {\n          this.recentQualityMatrix$.next(currentQm);\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    let pastDataSubscription: Unsubscribable | undefined;\n\n    const pastData$ = combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$]).pipe(\n      filter(\n        (data): data is [{ [key: string]: Array<string> }, { start: DateTime<true> }, void] =>\n          !!data[0] && !!data[1].start?.isValid\n      ),\n      throttleTime(500),\n      map(([categories, range]) => [categories, range.start] as const),\n      switchMap(([categories, start]) => this.loadPastData(categories, start)),\n      tap((pastQm) => {\n        this.pastQualityMatrix$.next(pastQm);\n      }),\n      takeUntilDestroyed(this.destroyRef)\n    );\n\n    this.isHistoricalDataEnabledFormField.valueChanges\n      .pipe(\n        tap((isEnabled) => {\n          if (isEnabled) {\n            pastDataSubscription = pastData$.subscribe();\n            this.categoryControls.setValue(this.categoryControls.value as { [key: string]: string[] }, {\n              emitEvent: true,\n            });\n            this.range.setValue(this.range.value as { start: DateTime; end: DateTime }, {\n              emitEvent: true,\n            });\n            this.refresh$.next();\n          } else {\n            pastDataSubscription?.unsubscribe();\n          }\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    this.refresh$.next();\n    this.isOnlySourcesWithMaterialEnabledFormField.setValue(this.isOnlySourcesWithMaterialEnabledFormField.value, {\n      emitEvent: true,\n    });\n  }\n\n  getTrend(row: QualityMatrixRow, column: QualityMatrixHeader): { delta?: number; trend?: string } {\n    if (this.pastQualityMatrix$.value === null || this.pastQualityMatrix$.value.columns.length === 0) {\n      return {};\n    }\n    const past_row = this.pastQualityMatrix$?.value.rows.find((pr) => pr.meta.id == row.meta.id);\n    const past_column = this.pastQualityMatrix$.value.columns.find((c) => c.id == column.id);\n    if (!past_row || !past_column) {\n      return {};\n    }\n\n    const delta = Math.floor(\n      (100 * row.counts[column.id].sufficient) / column.total -\n        (100 * past_row.counts[column.id].sufficient) / past_column.total\n    );\n\n    const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';\n    return { delta, trend };\n  }\n\n  getDonutSlices(row: QualityMatrixRow, column: QualityMatrixHeader): DonutSlice[] {\n    const sufficientTotal = row.counts[column.id].sufficient;\n    const insufficientTotal = row.counts[column.id].insufficient;\n    const missingTotal = column.total - sufficientTotal - insufficientTotal;\n    let sufficientFillRate = 0;\n    let insufficientFillRate = 0;\n    //Der Wert berechnet sich aus dem total_count - without_description_count.\n\n    if (column.total !== 0) {\n      sufficientFillRate = Math.round((100 * sufficientTotal) / column.total);\n      insufficientFillRate = Math.round((100 * insufficientTotal) / column.total);\n    }\n    const rest = 100 - (sufficientFillRate + insufficientFillRate); // a.k.a. \"missing\"\n    return [\n      {\n        id: 3,\n        percent: rest,\n        color: '#D8EDFC',\n        label: `${rest} % (${missingTotal})`,\n        verboseLabel: `${missingTotal} Metadatens\\u00e4tze fehlen`,\n      },\n      {\n        id: 1,\n        percent: insufficientFillRate,\n        color: '#C20808',\n        label: `${insufficientFillRate} % (${insufficientTotal})`,\n        verboseLabel: `${insufficientTotal} Metadatens\\u00e4tze sind unzureichend ausgef\\u00fcllt`,\n      },\n      {\n        id: 2,\n        percent: sufficientFillRate,\n        color: '#4ABEFF',\n        label: `${sufficientFillRate} % (${sufficientTotal})`,\n        verboseLabel: `${sufficientTotal} Metadatens\\u00e4tze sind gut ausgef\\u00fcllt`,\n      },\n    ];\n  }\n\n  showTooltip(row: QualityMatrixRow, column: QualityMatrixHeader, cell: HTMLElement) {\n    if (row.counts[column.id].sufficient + row.counts[column.id].insufficient === 0) {\n      return;\n    }\n    if (this.tooltips.has(cell)) {\n      this.tooltips.get(cell)!.cancelCloseAction();\n      return;\n    }\n\n    const tooltipRef = this.tooltipService.create(DonutChartTooltipComponent, {\n      closeTimeout: openCloseTimeout,\n      openTimeout: openCloseTimeout,\n      data: {\n        label: column.label,\n        data: transformDonutChartData(this.getDonutSlices(row, column)),\n      },\n      overlayConfig: {\n        hasBackdrop: false,\n        positionStrategy: this.overlay.position().flexibleConnectedTo(cell).withPositions(centerPositionStrategy),\n        disposeOnNavigation: true,\n        panelClass: 'donut-chart-tooltip',\n      },\n      onClose: () => {\n        this.tooltips.delete(cell);\n      },\n    });\n    tooltipRef.open();\n    this.tooltips.set(cell, tooltipRef);\n  }\n\n  hideTooltip(cell: HTMLElement) {\n    this.tooltips.get(cell)?.markForClose();\n  }\n\n  findHorizontalScrollContainer(element: Element): Element | null {\n    if (element.scrollWidth > element.clientWidth && ['auto', 'scroll'].includes(getComputedStyle(element).overflowX)) {\n      // find the nearest horizontally scrollable element\n      return element;\n    } else {\n      if (!element.parentElement) {\n        return document.scrollingElement;\n      }\n      return this.findHorizontalScrollContainer(element.parentElement);\n    }\n  }\n\n  scrollToRight() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    const lastVisible = this.scrollMarkers.findLast(\n      (marker) => (marker.element.nativeElement as HTMLElement).getBoundingClientRect().left < viewportWidth\n    );\n\n    if (!lastVisible) {\n      return;\n    }\n\n    const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;\n\n    scrollingElement.scrollBy({\n      left: (lastVisible.element.nativeElement as HTMLElement).getBoundingClientRect().left - leftColumnWidth,\n      behavior: 'smooth',\n    });\n  }\n\n  scrollToLeft() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;\n\n    const firstVisible = this.scrollMarkers.find(\n      (marker) => (marker.element.nativeElement as HTMLElement).getBoundingClientRect().right > leftColumnWidth\n    );\n\n    if (!firstVisible) {\n      return;\n    }\n\n    scrollingElement.scrollBy({\n      left: (firstVisible.element.nativeElement as HTMLElement).getBoundingClientRect().right - viewportWidth,\n      behavior: 'smooth',\n    });\n  }\n\n  @HostListener('window:scroll')\n  @HostListener('window:resize')\n  onScroll() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    this.scrollLeft.set(scrollingElement.scrollLeft);\n    this.scrollWidth.set(scrollingElement.scrollWidth);\n  }\n}\n","<mat-card appearance=\"raised\">\n  <mat-card-header *ngIf=\"pageTitle\">\n    <mat-card-title>Qualitätsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n  </mat-card-header>\n  <mat-card-content class=\"toolbar\">\n    <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n                       [options]=\"categoryFilterValues.get(filter.key)\"\n                       [inputFormControl]=\"filter.value\"\n                       [label] = \"'filter.' + filter.key | translate\" />\n    <div class=\"actionbar\">\n      <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n        show only sources with material\n      </mat-slide-toggle>\n      <div style=\"flex: 1 1 auto\"></div>\n      <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n      <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n        show historical data\n      </mat-slide-toggle>\n      <button\n        mat-icon-button\n        color=\"primary\"\n        [disabled]=\"isLoading()\"\n        (click)=\"refresh()\"\n        matTooltip=\"Aktualisiere den IST-Stand\"\n        matTooltipShowDelay=\"500\"\n      >\n        <mat-icon>refresh</mat-icon>\n      </button>\n      <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n      <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n        <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n          <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n        </button>\n        <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n          <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n        </button>\n      </div>\n    </div>\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]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Row Header Column -->\n    <ng-container matColumnDef=\"row-header\" sticky>\n      <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        matTooltip=\"{{row.meta.alt_label}}\"\n        class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n      >\n        {{\"quality_matrix.\" + row.meta.label | translate }}\n      </td>\n    </ng-container>\n    <!-- one column for the source -->\n    <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n      <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n    </ng-container>\n    <!-- /source -->\n    <!-- current Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n      <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n      <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n          <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n            <div>\n              <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n            </div>\n          </a>\n          <ng-template #emptyDiv>\n            <div>-</div>\n          </ng-template>\n      </td>\n    </ng-container>\n    <!-- /current Data Columns -->\n    <!-- past Data Columns -->\n    <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n      <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n        <th class=\"historic-data-cell\" mat-header-cell\n            *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n          <br>{{ getPastColumn(column.id)?.total }}\n        </th>\n        <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n          <ng-container *ngIf=\"getTrend(row, column) as trend\">\n            <span [ngClass]=\"trend.trend\">\n              <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n              {{ trend.delta }}\n            </span>\n          </ng-container>\n\n        </td>\n      </ng-container>\n    </ng-container>\n    <!-- /past Data Columns -->\n\n      <!-- Generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n    <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n  </table>\n</mat-card>\n"]}
@@ -1,17 +0,0 @@
1
- import { Directive, ElementRef, inject } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- export class ScrollMarkerDirective {
4
- constructor() {
5
- this.element = inject(ElementRef);
6
- }
7
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollMarkerDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
8
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: ScrollMarkerDirective, isStandalone: true, selector: "[metaqs2ScrollMarker]", ngImport: i0 }); }
9
- }
10
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollMarkerDirective, decorators: [{
11
- type: Directive,
12
- args: [{
13
- selector: '[metaqs2ScrollMarker]',
14
- standalone: true,
15
- }]
16
- }] });
17
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Nyb2xsLW1hcmtlci5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1tZXRhLXdpZGdldHMtbGliL3NyYy9saWIvY29tcG9uZW50cy9xdWFsaXR5LW1hdHJpeC9zY3JvbGwtbWFya2VyLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBTTlELE1BQU0sT0FBTyxxQkFBcUI7SUFKbEM7UUFLUyxZQUFPLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0tBQ3JDOytHQUZZLHFCQUFxQjttR0FBckIscUJBQXFCOzs0RkFBckIscUJBQXFCO2tCQUpqQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSx1QkFBdUI7b0JBQ2pDLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgRWxlbWVudFJlZiwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ1ttZXRhcXMyU2Nyb2xsTWFya2VyXScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG59KVxuZXhwb3J0IGNsYXNzIFNjcm9sbE1hcmtlckRpcmVjdGl2ZSB7XG4gIHB1YmxpYyBlbGVtZW50ID0gaW5qZWN0KEVsZW1lbnRSZWYpO1xufVxuIl19